使用kotlin编写html dsl框架

前排提醒,这个框架就是我写着玩的,如果您已经会使用vue或其他前端框架,这篇文章可能对您没有什么意义。即使您不会如上提到的框架,也不要对该框架报有过高的期待,该框架更多的是,我自己的自娱自乐。

这里还要提醒一下,该框架没有实现对css和js的支持,就是一个生成html代码的工具。

前言

为什么我要写这个玩意出来?因为我有时想用网页写一写游戏评测文章,而用html编写就可以比较方便地通过浏览器在不同网页之间跳转。如果编写markdown文章或其他方式,就比较麻烦了。
而我作为一名安卓开发者,对前端开发并不熟悉,而且对网页的要求也不高。只需要可显示不同样式的文字、图片和链接等功能,所以就不想花精力去学习前端框架。当然了,以后如果工作需要,那去学习也不可厚非。
编写该框架确实有如上的原因,但还有一个原因:我想试试看我能不能写出来。所以就经历了框架设计、改进框架不合理的代码等过程,最后有了现在代码。

使用和演示

上面说了那么多,接下来贴一下使用代码。

html {
    header {
        title("test")
    }
    body {
        div {
            a {
                href = "https://www.baidu.com"
                text = "baidu"
            }
            h1("这是h1")
            text("text")
            
        }
        text("text")
    }
}.toHtmlCode().also(::println)

// println
<html><header><title>test</title></header><body><div><a href="https://www.baidu.com">baidu</a><h1>这是h1</h1>text<img/></div>text</body></html>

可以看到,只需很简单的编写代码,就能够输出完整的html代码,而不需要手写很多标签代码。
再看看我使用该框架编写html的实际代码和最终效果。

在这里插入图片描述
在这里插入图片描述

在这里插入图片描述

代码

上面代码看似简单,但实际上,调用的代码一点都不少,想看实现的代码,可以直接看这个github链接,接下来是代码解析

HtmlTag
首先,需要一个类来描述html代码,所以我就创建了HtmlTag,任何一个标签都是HtmlTag的实现类。
interface HtmlTag {
    fun getTagString(): String

    // 获取标签的属性,如id、width、height等
    // Pair的first就是属性的字符串,second就是属性的值,类型设计为Any是考虑到Int、Double这些类型。如果限制为String,那每次都需要手动调用一次toString,这样做其实挺麻烦的
    fun getAttributeList(): List<Pair<String, Any>>

    // 转换成html代码,每个标签做好自己的转换任务,由上级调用下级的该方法进行转换,最终形成一条调用链,这样就能非常方便地生成html代码
    fun toHtmlCode(): String
}

从上面的演示代码可以看到,还有html这个方法,这个方法获取到的就是一个HtmlRoot对象,看看该对象里面有什么代码

class HtmlRoot: HtmlTag {
    var header: HtmlHeaderRoot? = null
    var body: HtmlBodyRoot? = null

    override fun getTagString(): String = TAG

    override fun toHtmlCode(): String {
        return generateHtmlCode(listOfNotNull(header, body))
    }

    override fun getAttributeList(): List<Pair<String, Any>> = emptyList()

    companion object {
        const val TAG = "html"
    }
}

这里的generateHtmlCode是一个扩展方法,下面会提到,先把注意力放到其他地方。

HtmlBody

可以看到,该类里面,有header和body对象,header和body的代码类似,我就把body拿出来讲。
不过在看看body的代码之前,先了解一下body的基类。
HtmlBody

interface HtmlBody: HtmlTag

就一个非常简单的接口,所有和body有关的代码,都必须为该接口的实现类,包括HtmlBodyRoot本身。该类有两个直接实现类:HtmlBodySingleHtmlBodyGroup
HtmlBodySingle

abstract class HtmlBodySingle<T: HtmlBody>: HtmlBody {
    open var body: T? = null
}

HtmlBodyGroup

abstract class HtmlBodyGroup<T: HtmlBody>: HtmlBody {
    protected open var internalBodyList: MutableList<T> = ArrayList()

    open fun addHtmlBody(body: T) {
        internalBodyList.add(body)
    }

    open fun addAllHtmlBody(list: List<T>) {
        internalBodyList.addAll(list)
    }

    open fun removeHtmlBody(body: T) {
        internalBodyList.remove(body)
    }

    open fun clearBodyList() {
        internalBodyList.clear()
    }

    open fun getBodyList(): List<T> {
        return internalBodyList
    }
}

可以看到,single只能有一个body,而group可以有多个body。为什么要这样做?因为我考虑到有些标签只能有一个子标签,如a、h这些标签,而有一些则可以多个,如body、div等。
一开始,我考虑的是,将single作为一个特殊的group标签,做一些实现,保证开发者永远只能在list里面添加一个标签,就像android的ScrollView一样。但随后想了想,还是不要这样做,因为这样会增加开发者出错的几率。
上面还有泛型这个东西,我考虑的是,有一些标签的子标签类型不需要那么宽泛,只需某些特定的标签,如table、ul、ol等,所以我就加了一个泛型上去。
现在再看看body标签的代码

class HtmlBodyRoot: HtmlBodyGroup<HtmlBody>() {
    var style: HtmlStyleRoot? = null

    override fun getTagString(): String = TAG

    // 这里就是将style和bodyList转换成String,生成一个字符串List,传递给generateHtmlCodeByStringList方法去生成html代码
    override fun toHtmlCode(): String {
        val style = style?.toStyleCode()?.takeIf { it.isNotEmpty() } ?: ""
        return generateHtmlCodeByStringList(listOf(style, getBodyList().bodyTagListToString()))
    }

    // 我的印象里,body好像没有属性,所以我直接返回一个空List
    override fun getAttributeList(): List<Pair<String, Any>> = emptyList()

    companion object{
        const val TAG = "body"
    }
}

fun <T: HtmlBody> List<T>.bodyTagListToString() = joinToString("") { it.toHtmlCode() }

可以看到,body的代码也没什么,就是一些很普通的代码。
我贴出来的代码有不少调用generateHtmlCode这个方法,来看看该方法的实现代码。

fun HtmlTag.generateHtmlCode(): String {
    return generateHtmlCode("")
}

// 大部分body会调用这里的single和group的扩展方法,可以看到,这里的代码最终都会调用toHtmlCode方法
fun <T: HtmlBody> HtmlBodySingle<T>.generateHtmlCode(): String {
    return generateHtmlCode(body?.toHtmlCode() ?: "")
}

fun <T: HtmlBody> HtmlBodyGroup<T>.generateHtmlCode(): String {
    return generateHtmlCode(getBodyList().bodyTagListToString())
}

fun <T: HtmlHeader> HtmlHeaderGroup<T>.generateHtmlCode(): String {
    return generateHtmlCode(getHeaderList().headerTagListToString())
}

fun <T: HtmlTag> HtmlTag.generateHtmlCode(htmlList: List<T>): String {
    return generateHtmlCode(htmlList.tagListToString())

}

// HtmlBodyRoot就是调用这个方法生成html代码
fun HtmlTag.generateHtmlCodeByStringList(codeList: List<String>): String {
    // 这里也只是将String List调用joinToString转换成一个String而已
    return generateHtmlCode(codeList.joinToString(""))
}

// 所有代码最终都会调用该方法,可以直接看该方法的代码
fun HtmlTag.generateHtmlCode(value: String): String {
    // 这里就会调用HtmlTag的getAttributeList方法,将它们转换成first="second"这样的代码,并在转换后的字符串前面加一个空格。如果该List为空,就直接返回空字符串。
    // 为什么要加一个空格,因为如果不加空额,标签代码就会和属性代码贴在一起。如:<ahref=""/>
    val attributeString = getAttributeList().takeIf { it.isNotEmpty() }?.joinToString(" ") {
        "${it.first}=\"${it.second}\""
    }?.let { " $it" } ?: ""
    // 如果value为空,就生成闭标签的代码,否则就生成开标签的代码
    return if(value.isEmpty()) {
        "<${getTagString()}$attributeString/>"
    } else {
        buildString {
            // 配合上面的single和group的generateHtmlCode扩展方法,就不难理解
            // 将子body的代码包在自己的标签里面,而子body也会调用自己的子body包在自己的标签里面
            // 最后一层包一层,就可以输出一串复杂的html代码
            append("<${getTagString()}$attributeString>")
            append(value)
            append("</${getTagString()}>")
        }
    }
}

为什么要将它们作为扩展方法,而不是放到HtmlTag里面?因为这里面有一些方法不是每个实现类都用得上。如果将这些代码放到顶层接口里面,最终会导致每个实现类生成的class文件里面,有很多用不上的代码,所以将这些代码作为扩展方法是比较实际的。而且接口更多是起到定义规范的作用,而不是当工具类使用。

getAttributeList

再提一下getAttributeList的实现,拿a标签举例
HtmlBodyGeneralAttribute.kt

// 属性的根类是HtmlBodyGeneralAttribute,这里面包含了所有基础标签,不过我只是将开发中需要的属性加上去,后续如果还需要其他属性,可以自己加
interface HtmlBodyGeneralAttribute<T: HtmlBodyGeneralAttributeEntity> {
    // 为属性提供默认实现,这样做了之后,实现类就不用手动写这些代码了
    // 这里的attributeEntity就是存放这些属性的实体,如果在这里直接给这个字段编写get方法,每次调用该字段都会重写new一个对象
    // 所以只能交给实现类去new,但也仅仅需要编写这一行代码,所以我认为这没有什么负担
    val attributeEntity: T

    // 通过字段的形式来设置属性,这样外部用起来就比较方便,而不用去调用方法
    // 这些字段默认是否为null,就自己判断了,如果觉得不需要null,也可以将?去掉
    var id: String?
        get() = attributeEntity.id
        set(value) {
            attributeEntity.id = value
        }

    var width: String?
        get() = attributeEntity.width
        set(value) {
            attributeEntity.width = value
        }

    var height: String?
        get() = attributeEntity.height
        set(value) {
            attributeEntity.height = value
        }

    // 最后通过该方法生成一个Pair List
    // toPairByStringValue的代码已经放在下面了,该方法会判断String Value是否为空字符串,如果是,就烦恼会一个空的Pair对象
    // 这里调用listOfNotNull,所以空的Pair对象就会直接被过滤掉
    fun getAttributeList(): List<Pair<String, Any>> {
        return listOfNotNull(
            HtmlBodyAttribute.general.ID toPairByStringValue attributeEntity.id,
            HtmlBodyAttribute.general.WIDTH toPairByStringValue attributeEntity.width,
            HtmlBodyAttribute.general.HEIGHT toPairByStringValue attributeEntity.height,
        )
    }
}

open class HtmlBodyGeneralAttributeEntity {
    var id: String? = null
    var width: String? = null
    var height: String? = null
}

infix fun String.toPairByStringValue(value: String?): Pair<String, String>? {
    return value?.takeIf { it.isNotEmpty() }?.let { this to value }
}

HtmlBodyATag.kt

// 这个类用于存放a标签需要的属性,该类需要继承HtmlBodyGeneralAttribute,并提供attibuteEntity对象类型
// 所以就写一个a的attibuteEntity继承GenernalAttributeEntity
interface HtmlBodyAAttribute: HtmlBodyGeneralAttribute<HtmlBodyAAttributeEntity> {
    var href: String?
        get() = attributeEntity.href
        set(value) {
            attributeEntity.href = value
        }

    var target: String?
        get() = attributeEntity.target
        set(value) {
            attributeEntity.target = value
        }

    // 重点是这里,继承之后,需要重写该方法,将super的结果取出来,并放一个新的List,形成一个二维List
    // 最后再调用list的flatten方法, 将二维List变成一维
    // 当然了,如果觉得这种实现方法不太好,也可以自己换一种更好的实现方式
    override fun getAttributeList(): List<Pair<String, Any>> {
        val superList = super.getAttributeList()
        val currentList = listOfNotNull(
            HtmlBodyAttribute.a.HREF toPairByStringValue attributeEntity.href,
            HtmlBodyAttribute.a.TARGET toPairByStringValue attributeEntity.target,
        )
        return listOf(superList, currentList).flatten()
    }
}

class HtmlBodyAAttributeEntity: HtmlBodyGeneralAttributeEntity() {
    var href: String? = null
    var target: String? = null
}

// 实现HtmlBodyAAttribute接口
class HtmlBodyATag: HtmlBodySingle<HtmlBody>(), HtmlBodyAAttribute {
    // 在a标签里面,只需重写这个字段,其他的都不用做,就拥有了设置id、width、href等功能
    override val attributeEntity: HtmlBodyAAttributeEntity = HtmlBodyAAttributeEntity()

    override fun getTagString(): String = TAG

    override fun toHtmlCode(): String {
        return generateHtmlCode()
    }
    
    // 这里需要override是没办法的事情,因为HtmlTag本身也有一个名称一样的方法,并且没有提供实现
    // 这里调用super就可以将AAttribute的实现直接拿过来用,所以
    override fun getAttributeList(): List<Pair<String, Any>> {
        return super.getAttributeList()
    }

    companion object{
        const val TAG = "a"
    }
}
纯文本

从上面的HtmlBody、HtmlBodySinge和HtmlBodyGroup的代码可以看到,没有一个属性用来编写纯文本,但html想要编写纯文本的代码,只需在空白部分编写就可以显示出来。
鉴于这种情况,我编写了HtmlBodyTextTag来实现这个功能,代码非常简单

class HtmlBodyTextTag: HtmlBody {
    var text: String = ""

    override fun getTagString(): String = ""

    override fun toHtmlCode(): String {
        return text
    }

    override fun getAttributeList(): List<Pair<String, Any>> = emptyList()
}

如果某个标签需要纯文本的内容,就可以设置这样一个body

dsl代码

上面将body的基本架构都介绍完了,接下来写一下dsl的代码是怎么写的
htmlDSL.kt

inline fun html(action: HtmlRoot.() -> Unit): HtmlRoot {
    return HtmlRoot().also {
        it.action()
    }
}

inline fun HtmlRoot.header(action: HtmlHeaderRoot.() -> Unit): HtmlHeaderRoot {
    return HtmlHeaderRoot().also {
        it.action()
        header = it
    }
}

inline fun HtmlRoot.body(action: HtmlBodyRoot.() -> Unit): HtmlBodyRoot {
    return HtmlBodyRoot().also {
        it.action()
        body = it
    }
}

代码还是比较简单的,想要写注释,但不知道要写什么
有关body dsl的代码还是有点多的,所以拿一部分出来讲
htmlBodyGenenralDSL.kt

// 给single扩展方法之后,在single里面就可以直接调用a了,而不需要用body = getA这种麻烦的形式
inline fun HtmlBodySingle<HtmlBody>.a(action: HtmlBodyATag.() -> Unit) {
    getA.also {
        it.action()
        body = it
    }
}

// 由于HtmlBodyRoot也是一个HtmlBodyGroup对象,所以这样写了之后,就能为body添加一个a标签
// 返回类型:返回自己本身。为什么要这样做?因为返回自己之后,就能链式调用代码
// 比如写一个a标签之后,如果不想换行去写一个text标签,就可以直接在a标签的代码后面.去调用,否则需要手动写";"才能调用
inline fun HtmlBodyGroup<HtmlBody>.a(action: HtmlBodyATag.() -> Unit): HtmlBodyGroup<HtmlBody> {
    getA.also {
        it.action()
        addHtmlBody(it)
    }
    return this
}

htmlBodyGenenralGetDSL.kt

inline val getA: HtmlBodyATag
    get() = HtmlBodyATag()

给group扩展之后,就不只是body可以使用,像div、li、th、td这些,也都可以通过这种方法add一个body,使用起来非常方便。
从上面还能看到,有一个getA,为什么要这样做?
可以想象一下,如果外部需要new一个A标签的对象,那就需要记住A标签的对象名称,但名称又那么长,每次写起来都挺麻烦的。而如果用这种形式,就可以让开发者无需记住对象名称,需要使用时,只需在标签前面加个get就可以拿到对应的标签对象,这不是很方便吗?
大部分标签会提供get方法,并放到GetDSL里面。
text

inline fun HtmlBodySingle<HtmlBody>.text(text: String){
    getText.also {
        it.text = text
        body = it
    }
}

var HtmlBodySingle<HtmlBody>.text: String
    get() = (body as? HtmlBodyTextTag)?.text ?: ""
    set(value) {
        val body = body as? HtmlBodyTextTag ?: getText.also {
            body = it
        }
        body.text = value
    }

inline fun HtmlBodyGroup<HtmlBody>.text(text: String): HtmlBodyGroup<HtmlBody> {
    getText.also {
        it.text = text
        addHtmlBody(it)
    }
    return this
}

text除了提供2个方法,还为single提供了text这个字段,这样如果想要给某个标签设置text,就可以通过这种方式
dsl的代码基本都是a、text标签这种形式,其他代码都是大同小异,所以其他代码我就不贴出来了。

字数统计
最后补充一个字数统计的功能,如果要统计一个网页的字数,可以使用js进行统计,但该框架没有提供js相关的api,所以写起来有点麻烦,自然地,我也不打算用js去统计字数。
除了js,我还想到用正则表达式,将html的文本找出来,但试了几个正则表达式,我都没能将文本找出来。
最后就想到,直接提取body里面所有text对象,并获取text对象的文本内容,最后计算出字数。
fun HtmlRoot.getTextLength(): Int {
    // 通过递归调用,就可以获取到所有text对象,获取完成后,就可以通过text里面的text字段计算字数
    return body?.getTextBodyList()?.sumOf {
        it.text.length
    } ?: 0
}

fun HtmlBody.getTextBodyList(): List<HtmlBodyTextTag>? {
    return when(this) {
        is HtmlBodyTextTag -> {
            arrayListOf(this)
        }
        is HtmlBodySingle<*> -> {
            getTextBodyList()
        }
        is HtmlBodyGroup<*> -> {
            getTextBodyList()
        }
        else -> null
    }
}

fun HtmlBodyGroup<*>.getTextBodyList(): List<HtmlBodyTextTag>? {
    val list = ArrayList<HtmlBodyTextTag>()
    getBodyList().forEach {
        // 这里就会调用上面的HtmlBody的getTextBodyList方法
        // 如果获取到一个空对象,就不会addAll到list里面
        it.getTextBodyList()?.also(list::addAll)
    }
    // 如果list不为空,就返回该List,否则就返回null
    return list.takeIf { it.isNotEmpty() }
}

fun HtmlBodySingle<*>.getTextBodyList(): List<HtmlBodyTextTag>? {
    return body?.getTextBodyList()
}

我自己没有想出来如何用正则表达式统计字数,所以抱着试一试的想法问了chatGPT,想不到还真得到我想要的代码。而且比我想象中的简单

在这里插入图片描述

fun getTextByHtmlCode(html: String): String {
    // 移除 HTML 标签
    var text = html.replace("\\<.*?\\>".toRegex(), "")
    // 移除 HTML 转义字符
    text = text.replace("&.*?;".toRegex(), "")
    // 移除多余空格和换行符
    text = text.trim { it <= ' ' }.replace("[\\s\\t]+".toRegex(), " ")
    return text
}

可以看到,直接暴力匹配<>这个括号就行,不管里面有什么内容。而且我也用一段5000多文本的html页面测试过了,得到的结果时一样的,所以代码是不存在什么问题的。只不过执行效率我就不清楚了,我不知道正则表达式的替换效率是怎么样的。而我提供的代码,本质是递归调用,也好不到哪去。
从chatGPT发送的代码可以看到,最后还会将字符串转换成字符数组,这段在我看来是没必要的,所以我就去掉了。

后记

上面简单地讲解了该框架的使用方法和实现方式,其他代码麻烦看看github里面的代码。
由于该框架主要是为了方便自己,所以提供的标签和属性都不全,如果自己需要哪个标签,可以根据我编写的代码,自己做一套实现。模板代码已经写出来了,剩下的就是体力活了。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Kotlin 支持的框架有:Ktor、Spring、Anko、Kodein、Kotlinx.html、Spek、Kotlin Android Extensions、Kotlin Serialization、Kotlin Coroutines 和 TornadoFX。 ### 回答2: Kotlin是一种基于Java虚拟机的编程语言,因此可以利用Java生态圈中的众多框架。此外,Kotlin也有一些独有的框架,下面是一些常见的Kotlin框架: 1. Ktor: Ktor是一个轻量级的Web框架,用于构建异步和非阻塞的Web应用程序。它提供了一个简单易用的API,支持各种服务器和客户端功能。 2. Anko: Anko是Kotlin的一个功能强大且易于使用的库,用于简化Android应用程序的开发。它提供了一系列的DSL(领域特定语言)来简化UI创建,数据库操作,异步任务等常见任务。 3. Exposed: Exposed是一个轻量级的ORM(对象关系映射)库,用于简化数据库的访问和操作。它提供了简洁的API,易于理解和使用,支持各种数据库。 4. Koin: Koin是一个轻量级的依赖注入框架,用于管理应用程序中的对象和它们之间的依赖关系。它通过提供简单直观的API,使得依赖注入变得容易。 5. TornadoFX: TornadoFX是一个用于构建JavaFX应用程序的Kotlin框架。它提供了强大且易于使用的API,通过DSL的方式简化了UI创建和事件处理。 这只是Kotlin的一些框架示例,实际上还有很多其他的框架可供选择,涵盖了各种应用程序开发的需求。Kotlin作为一种灵活且功能强大的语言,为开发人员提供了丰富的选择。 ### 回答3: Kotlin是一种基于Java虚拟机的编程语言,因其简洁、安全和互操作性等特点,逐渐在开发者中得到广泛使用Kotlin具备强大的框架生态系统,以下是其中一些流行的框架: 1. Ktor:Ktor是一个轻量级的Web框架,用于构建可扩展且异步的后端应用程序。它提供简洁的API,易于使用和学习,并支持异步协程。 2. Spring Boot with KotlinKotlin集成了Spring Framework,使得使用Kotlin编写Spring Boot应用程序变得更加简洁和易用。Spring Boot提供了丰富的功能和工具来快速构建和部署应用程序。 3. Exposed:Exposed是一个轻量级的SQL库,用于与数据库交互。它使用Kotlin的强类型和DSL(领域特定语言)的特性,提供了简单、类型安全且易于维护的数据库访问方式。 4. Anko:Anko是一个用于Android开发的Kotlin库,它提供了简化和加快Android应用程序开发的工具和实用功能,例如DSL布局构建、数据库操作和异步任务处理。 5. Arrow:Arrow是一个函数式编程库,旨在帮助开发者以函数式和声明式的方式构建应用程序。它提供了一组操作符和类型类, 帮助开发者编写简洁、可维护和高效的代码。 除了上述框架外,Kotlin还支持许多其他的框架,例如Koin(轻量级的依赖注入框架)、JUnit(单元测试框架)和Mockito(模拟框架)等。这些框架使得Kotlin在不同领域和平台上开发应用程序时更加便捷和高效。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值