jjzjj

开源|用 Java 实现一个生成 Markdown 文本的工具

Elltor 2024-01-05 原文

公司的 IM 每天有许多机器人推送的消息,我也在使用,这个功能是好的,但是当我们想去发送一些格式优美的消息时,却要费许多功夫,主要来源于字符串拼接,如果要拼接出 Markdown 格式的那就更费力了,另外由拼接带来的是混乱的代码,为了解决这个痛点,我写了一个 Java 代码生成 Markdown 文本的工具,还给它起了一个酷名字。

MdKiller —— Markdown 杀手 🥷。

MdKiller 是一个格式化生成 Markdown 文本的工具,支持常用 Markdown 格式生成,例如引用块、代码块、有无序列表、表格等,内容上支持字体样式(style)和内容的嵌套,适用于 IM 消息 Markdown 排版。

以下为 Github 仓库 介绍页内容。

一、使用

1、引入依赖,由于是单文件,直接把文件拷贝到项目即可使用。

2、使用,示例如下:

@Test
public void test(){
    String md=MdKiller.of()
    .title("一个标题")
    .text("文本")
    .ref()
        .text("文本1")
        .text("文本2")
        .ul()
            .text("文本3")
            .text("文本4")
        .endUl()
    .endRef()
    .link("链接","https://elltor.com")
    .build();
    System.out.println(md);
}

输出 Markdown 文本:

### 一个标题
文本
> 文本1
> 文本2
> - 文本3
> - 文本4

[链接▸](https://elltor.com)

P.S. 更多演示可以参考单元测试。

二、特性介绍

  • 易用性:易于排版文案、支持主流 Markdown 语法内容,就像用代码写 Markdown。
  • 通用性:生成内容通用,工具生成标准 Markdown 格式并保障最大兼容性。
  • 扩展性:具备一定扩展性,扩展 API 简单;内容多样,理论上所有输入的文本均支持样式。
  • 集成:除 JDK 外无依赖;小巧,单文件约 700+ 行代码;
  • 鲁棒性:具备一定容错性,所有接口均经过单元测试,见目录 test/java/com/elltor/md

三、最佳实践

在 IM 中排版消息是痛苦的,这个工具就是要解决这个问题,通常在 IM 排版消息会遇到下列问题:

  1. 在程序中直接拼接很难对 Markdown 进行排版,拼接使人头大,维护更加麻烦。(P.S. 不反对使用模版,但当增加/删除模版中的参数时也是麻烦的)
  2. 差的 Markdown 消息特点:文案没有结构、重点不突出、交互差,不能突出主题。

因此,我们改善 IM 消息实际上就是解决上面两个问题。

3.1 链式调用 vs 普通调用

用不同的风格生成 Markdown 文本。

/**
  * 链式调用 vs 普通调用
  */
@Test
public void callByChainShow() {
    String md = MdKiller.of()
            .title("标题")
            .text("文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落")
            .ref()
                .text("引用中的普通文本")
                .text("引用中的普通文本-设置颜色", MdKiller.Style.RED)
                .text("引用中的普通文本-加粗", MdKiller.Style.BOLD)
                .text("名字", Arrays.asList("值1", "值2", "值3","值4", "值5"))
            .endRef()
            .link("有问题点链接", "https://elltor.com")
            .build();
    System.out.println(md);
}

@Test
public void callByNormalShow() {
    MdKiller.SectionBuilder bd = MdKiller.of();
    bd.title("标题");
    bd.text("文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落");
    // 进入块级元素返回新 builder 对象,需要对象接收
    bd = bd.ref();
    bd.text("引用中的普通文本");
    bd.text("引用中的普通文本-设置颜色", MdKiller.Style.RED);
    bd.text("引用中的普通文本-加粗", MdKiller.Style.BOLD);
    bd.text("名字", Arrays.asList("值1", "值2", "值3","值4", "值5"));
    // 出块级元素返回父 builder 对象,需要对象接收
    bd = bd.endRef();
    bd.link("有问题点链接", "https://elltor.com");
    String md = bd.build();
    System.out.println(md);
}

输出:

### 标题
文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落
> 引用中的普通文本
> <font color='red'>引用中的普通文本-设置颜色</font>
> **引用中的普通文本-加粗**
> 名字:值1 | 值2 | 值3 | 值4 | 值5

[有问题点链接▸](https://elltor.com)

3.2 一个简洁消息模版

/**
  * 通过标题、引用中的文本和超链接形成了一个较为有格式的排版
  */
@Test
public void aBeautifulMsgTemplate() {
    String md = MdKiller.of()
            .title("标题")
            .text("文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落")
            .ref()
                .text("引用中的普通文本")
                .text("引用中的普通文本-设置颜色", MdKiller.Style.RED)
                .text("引用中的普通文本-加粗", MdKiller.Style.BOLD)
                .text("") // 一个空行
                .text("名字", Arrays.asList("值1", "值2", "值3","值4", "值5"))
            .endRef()
            .link("有问题点链接", "https://elltor.com")
            .build();
    System.out.println(md);
}

输出:

### 标题
文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落
> 引用中的普通文本
> <font color='red'>引用中的普通文本-设置颜色</font>
> **引用中的普通文本-加粗**
>
> 名字:值1 | 值2 | 值3 | 值4 | 值5

[有问题点链接▸](https://elltor.com)

3.3 用列表使内容更清晰

/**
  * 通过标题、列表、超链接形成排版
  */
@Test
public void aBeautifulMsgTemplate2() {
    String md = MdKiller.of()
            .title("标题")
            .text("文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落")
            .ul()
                .text("引用中的普通文本")
                .text("引用中的普通文本-设置颜色", MdKiller.Style.RED)
                .text("引用中的普通文本-加粗", MdKiller.Style.BOLD)
                .text("名字", Arrays.asList("值1", "值2", "值3","值4", "值5"))
            .endUl()
            .link("有问题点链接", "https://elltor.com")
            .build();
    System.out.println(md);
}

输出:

### 标题
文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落 文本段落
- 引用中的普通文本
- <font color='red'>引用中的普通文本-设置颜色</font>
- **引用中的普通文本-加粗**
- 名字:值1 | 值2 | 值3 | 值4 | 值5

[有问题点链接▸](https://elltor.com)

3.4 使用表格

/**
  * 可以直接放置表格,或者在引用块中嵌套表格
  */
@Test
public void tableShow() {
    String[][] data = {
            {"姓名",  "姓别",  "芳龄", "身高"},
            {"不知火舞",null,   "18", "173"},
            {"孙策",  "男",    "23", "181"},
            {"李白",  "男",    null, "179"},
    };
    
    String md = MdKiller.of()
            .title("使用表格")
            .text("月老的记事本:", MdKiller.Style.RED)
            .table()
                .data(data)
            .endTable()
            .subTitle("嵌套表格")
            .ref()
                .text("月老的记事本:", MdKiller.Style.YELLOW)
                .table()
                    .data(data)
                .endTable()
            .endRef()
            .build();
    System.out.println(md);
}

输出:

### 使用表格
<font color='red'>月老的记事本:</font>

姓名 | 姓别 | 芳龄 | 身高
- | - | - | -
不知火舞 |  | 18 | 173
孙策 | 男 | 23 | 181
李白 | 男 |  | 179

#### 嵌套表格
> <font color='gold'>月老的记事本:</font>
> 
> 姓名 | 姓别 | 芳龄 | 身高
> - | - | - | -
> 不知火舞 |  | 18 | 173
> 孙策 | 男 | 23 | 181
> 李白 | 男 |  | 179

3.5 生成模版以提高性能

/**
  * 通过工具生成一个消息模版,你可以通过缓存消息模版进一步提高性能
  */
@Test
public void genTemplateAndCacheShow() {
    String mdTemplate = MdKiller.of()
            .title("%s")
            .text("%s")
            .ref()
                .text("%s")
            .endRef()
            .link("%s", "%s")
            .build();
    String realMd = String.format(mdTemplate,
            "消息模版标题",
            "这是一个消息模版,你可以通过缓存消息模版以提高性能。",
            "在引用中的消息 —— 这是一个消息模版,你可以通过缓存消息模版以提高性能。",
            "详情链接🔗", "https://elltor.com");
    System.out.println(realMd);
}

输出:

### 消息模版标题
这是一个消息模版,你可以通过缓存消息模版以提高性能。
> 在引用中的消息 —— 这是一个消息模版,你可以通过缓存消息模版以提高性能。

[详情链接🔗▸](https://elltor.com)

3.6 灵活在文本中拼接 Markdown 样式

@Test
public void joinMarkdownShow() {
    String md = MdKiller.of()
            .text("### 标题")
            .ref()
                .text("**文本**")
                .text("名字", "*值(斜体)*")
                .text("加粗文本", MdKiller.Style.BOLD)
                .link("链接", "https://elltor.com")
            .endRef()
            .text("[详情链接🔗](https://elltor.com)")
            .build();
    System.out.println(md);
}

输出:

### 标题
> **文本**
> 名字:*值(斜体)*
> **加粗文本**
> [链接▸](https://elltor.com)

[详情链接🔗](https://elltor.com)

四、最后

欢迎体验,如果有 Bug 可以直接在 GitHub 提 Issue,使用建议、交流 elltor(at)163.com,如果对你有帮助,可以点个免费的 Star。

有关开源|用 Java 实现一个生成 Markdown 文本的工具的更多相关文章

  1. ruby - 使用 RubyZip 生成 ZIP 文件时设置压缩级别 - 2

    我有一个Ruby程序,它使用rubyzip压缩XML文件的目录树。gem。我的问题是文件开始变得很重,我想提高压缩级别,因为压缩时间不是问题。我在rubyzipdocumentation中找不到一种为创建的ZIP文件指定压缩级别的方法。有人知道如何更改此设置吗?是否有另一个允许指定压缩级别的Ruby库? 最佳答案 这是我通过查看ruby​​zip内部创建的代码。level=Zlib::BEST_COMPRESSIONZip::ZipOutputStream.open(zip_file)do|zip|Dir.glob("**/*")d

  2. ruby - 使用 ruby​​ 将 HTML 转换为纯文本并维护结构/格式 - 2

    我想将html转换为纯文本。不过,我不想只删除标签,我想智能地保留尽可能多的格式。为插入换行符标签,检测段落并格式化它们等。输入非常简单,通常是格式良好的html(不是整个文档,只是一堆内容,通常没有anchor或图像)。我可以将几个正则表达式放在一起,让我达到80%,但我认为可能有一些现有的解决方案更智能。 最佳答案 首先,不要尝试为此使用正则表达式。很有可能你会想出一个脆弱/脆弱的解决方案,它会随着HTML的变化而崩溃,或者很难管理和维护。您可以使用Nokogiri快速解析HTML并提取文本:require'nokogiri'h

  3. ruby - 使用 Vim Rails,您可以创建一个新的迁移文件并一次性打开它吗? - 2

    使用带有Rails插件的vim,您可以创建一个迁移文件,然后一次性打开该文件吗?textmate也可以这样吗? 最佳答案 你可以使用rails.vim然后做类似的事情::Rgeneratemigratonadd_foo_to_bar插件将打开迁移生成的文件,这正是您想要的。我不能代表textmate。 关于ruby-使用VimRails,您可以创建一个新的迁移文件并一次性打开它吗?,我们在StackOverflow上找到一个类似的问题: https://sta

  4. ruby-on-rails - Rails - 一个 View 中的多个模型 - 2

    我需要从一个View访问多个模型。以前,我的links_controller仅用于提供以不同方式排序的链接资源。现在我想包括一个部分(我假设)显示按分数排序的顶级用户(@users=User.all.sort_by(&:score))我知道我可以将此代码插入每个链接操作并从View访问它,但这似乎不是“ruby方式”,我将需要在不久的将来访问更多模型。这可能会变得很脏,是否有针对这种情况的任何技术?注意事项:我认为我的应用程序正朝着单一格式和动态页面内容的方向发展,本质上是一个典型的网络应用程序。我知道before_filter但考虑到我希望应用程序进入的方向,这似乎很麻烦。最终从任何

  5. ruby-on-rails - 渲染另一个 Controller 的 View - 2

    我想要做的是有2个不同的Controller,client和test_client。客户端Controller已经构建,我想创建一个test_clientController,我可以使用它来玩弄客户端的UI并根据需要进行调整。我主要是想绕过我在客户端中内置的验证及其对加载数据的管理Controller的依赖。所以我希望test_clientController加载示例数据集,然后呈现客户端Controller的索引View,以便我可以调整客户端UI。就是这样。我在test_clients索引方法中试过这个:classTestClientdefindexrender:template=>

  6. ruby - 在 jRuby 中使用 'fork' 生成进程的替代方案? - 2

    在MRIRuby中我可以这样做:deftransferinternal_server=self.init_serverpid=forkdointernal_server.runend#Maketheserverprocessrunindependently.Process.detach(pid)internal_client=self.init_client#Dootherstuffwithconnectingtointernal_server...internal_client.post('somedata')ensure#KillserverProcess.kill('KILL',

  7. ruby - 如何使用 Ruby aws/s3 Gem 生成安全 URL 以从 s3 下载文件 - 2

    我正在编写一个小脚本来定位aws存储桶中的特定文件,并创建一个临时验证的url以发送给同事。(理想情况下,这将创建类似于在控制台上右键单击存储桶中的文件并复制链接地址的结果)。我研究过回形针,它似乎不符合这个标准,但我可能只是不知道它的全部功能。我尝试了以下方法:defauthenticated_url(file_name,bucket)AWS::S3::S3Object.url_for(file_name,bucket,:secure=>true,:expires=>20*60)end产生这种类型的结果:...-1.amazonaws.com/file_path/file.zip.A

  8. ruby-on-rails - 如果 Object::try 被发送到一个 nil 对象,为什么它会起作用? - 2

    如果您尝试在Ruby中的nil对象上调用方法,则会出现NoMethodError异常并显示消息:"undefinedmethod‘...’fornil:NilClass"然而,有一个tryRails中的方法,如果它被发送到一个nil对象,它只返回nil:require'rubygems'require'active_support/all'nil.try(:nonexisting_method)#noNoMethodErrorexceptionanymore那么try如何在内部工作以防止该异常? 最佳答案 像Ruby中的所有其他对象

  9. ruby - 为什么 SecureRandom.uuid 创建一个唯一的字符串? - 2

    关闭。这个问题需要detailsorclarity.它目前不接受答案。想改进这个问题吗?通过editingthispost添加细节并澄清问题.关闭8年前。Improvethisquestion为什么SecureRandom.uuid创建一个唯一的字符串?SecureRandom.uuid#=>"35cb4e30-54e1-49f9-b5ce-4134799eb2c0"SecureRandom.uuid方法创建的字符串从不重复?

  10. java - 等价于 Java 中的 Ruby Hash - 2

    我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/

随机推荐