字节工程师自行开发了基于IntelliJ的终极文档套件

简介: 众所周知,程序员最讨厌的四件事:写注释,写文档,别人不写注释,别人不写文档。因此,有必要找到降低文档编写和维护成本的方法。目前写技术文档的模式如下:

前言
众所周知,程序员最讨厌的四件事:写注释,写文档,别人不写注释,别人不写文档。因此,有必要找到降低文档编写和维护成本的方法。目前写技术文档的模式如下:

image.png

痛点总结有三个方面:

image.png

针对上述问题,我们的解决方案:

本地编辑,浏览工作收敛到IDE,提供身临其境的体验;

在文档和代码之间建立强关联,减少复制,提高联动性,提高文档触摸率;

代码和文档属于Git仓库,借助版本管理,避免因业务迭代而导致文档版本与代码不匹配;

制作可以将文档导出到在线的工具,浏览器可以随时访问;

方案总览
image.png

与原始模式相比,新方案可以做到完全脱离浏览器 / 文档编辑器,线上页面的同步完全交给定时触发的自动化部署。

图中橙色部分是方案的重点,按分工分为线下和线上两部分。职责如下:

线下:IDEAPlugin。

实现自定义语言的分析和分析;

预览器、编辑器提供文档内容;

提供一系列实用功能,相关代码和文档;

线上:Gradle/DokkaPlugin。

桥接,重用IDEPlugin语义分析,生成预览内容的能力;

扩展Dokarenderer,实现HTML和飞书文档的导出能力;

方案建设采用了许多有趣的技术,后面详细介绍。

线下效果
IDEAPlugin提供侧边栏和强大的编辑器。以下从编辑和浏览两个角度介绍。

编辑体验
假设源代码如下:

public class ClassA {

public static final String TAG = "tag";



ClassB b;



/**
 * method document here.
 *
 * @param params input string
 */

public static void invoke(@NotNull String params) {

    System.out.println("invoke method!");

    System.out.println("this is method body: " + params);

}



public ClassA() {

    System.out.println("create new instance!");

}



private static final class ChildClass {



    /**
     * This is a method from inner class.
     */

    void innerInvoke() {

        System.out.println("invoke method from child!");

    }

}

}
此效果是在文档中添加此类引用:

image.png

与复制、粘贴代码不同,新方案具有以下优点:

相关性更强,预览会随着代码片段的变化而变化;

易于重构、引用类名、方法名、字段名重命名时,文档内容会自动改变,防止引用失效;

更直观,编辑,浏览时可以更快地找到代码源;

输入流畅,补充能力提高;

浏览体验

image.png

新方案比普通Markdown更友好:

沉浸式使用,界面嵌入IDE,无需跳转到其他应用;

提到的源代码旁边有行标,点击一键查看文档;

文档浏览器支持与IDE一致的代码亮度,引用跳转;

线上效果
代码中文档会定期自动部署到远端。以一篇真实业务文档举例,HTML 部署到轻服务后长这样:

image.png

对应飞书的产物长如下:
image.png

这些线上页面主要面向非当前团队的读者,内容由 CI 定时同步,暂不提供跳转到 IDE 的能力。

技术实现
项目的架构如图所示:
image.png

考虑到用户体验部分主要呈现在IDEA(AndroidStudio)中,我们的技术栈选择基于InteliJ。按模块可分为三部分:

基建层

IDEA Plugin

Gradle / Dokka Plugin

通用逻辑(语言实现相关)封装在基建层,仅依赖 IntelliJ Core。相对于 IntelliJ Platform,IntelliJ Core 仅保留语言相关的能力,精简了 codeInsight、UI 组件等代码,被广泛用于 IntelliJ 各大产品中(包括图中的 Kotlin、Dokka 等)。

以下将介绍这三个主要模块。

基建
在整个方案中,基础设施是所有功能的基石,其核心能力是建立代码和文档之间的关联。在这里,我们设计了一套标记语言CodeRef,以满足以下需求:

语法简洁,结构与源代码一一对应;

准确的指向,即必须满足一对一的关系;

支持只保留声明(去除body),提高信噪比;

具有扩展性,便于后续迭代新功能;

Coderef语言并不复杂,采用类似Kotlin/Java的风格,用关键字、字符串、括号构成句子和代码块,代码块中的每个节点都有相应的源代码节点。下图是一个简单的示例,相应的关系用着色文字标记:

image.png

注:即使文档内容不改变,一旦图片中的源代码部分发生变化,相应的渲染效果也会实时改变,产生动态绑定效果。那么,如何实现动态绑定呢?大致分为以下三个步骤:

设计语法,编写语言实现;

结合现有能力(InteliJCore、KotlinPlugin)获取双边语法树,从而建立从文档节点到源代码节点的单向对应关系;

结合现有能力(MarkdownParser)生成用于渲染的文档文本;

语言基础实现
基于InteliJPlatform,实现自定义语言至少要做以下几件事:

编写BNF定义,描述语法;

Parser、Psielement接口、flex定义等。

基于生成的flex文件和JFlex生成lexer;

用Psitreutil等工具编写Mixin,实现PSI中声明的自定义方法;

BNF是一切的基础,每个定义和价值的选择都非常重要。一个小例子:

{

/* ...一些必要的 Context */

tokens = [

    /* ...一些 Token,转换为代码中的 IElementType */

    AT='@'

    CLASS='class'

]

/* ...一些规则 */

extends("class_ref_block|direct_ref|empty_ref") = ref

extends("package_location|class_location") = ref_location

extends("class_ref|method_ref|field_ref") = direct_ref

}

ref_location ::= package_location | class_location

package_location ::= AT package_def {

pin=2 // 只有 '@' 和 package_def 一起出现时,才把整个 element 视为 package_location

}

class_location ::= AT class_def {

pin=2 // 只有 '@' 和 class_def 一起出现时,才把整个 element 视为 class_location

}

direct_ref ::= class_ref | method_ref | field_ref | empty_ref {

methods = [ // 一些自定义的 method,需要在下面指定的 mixin class 中给出实现

    getNameStringLiteral

    getReferencedElement

    getOptionalArgs

]

mixin="com.bytedance.lang.codeRef.psi.impl.CodeRefDirectRefMixin"

}

class_ref ::= CLASS L_PAREN string_literal [COMMA ref_args_element*] R_PAREN {

methods = [

    property_value=""

]

pin=1 // 即遇到第一个元素 class 后,就将当前 element 匹配为 class_ref

}
上面的小片段中定义了 @class("")、@package("")、class("", ...) 语法。实战中比较关键的是 pin 和 recoverWhile,前者影响一段“未完成”的代码的类型,后者控制一段规则何时结束。具体参考 Grammar-Kit。

编写完成后,我们可以使用Grammar-Kit生成Parser和Lexer。前者负责最基本的语法亮度,后者负责输出PSI树。在自定义的ParserDefinition中注册两者,然后结合自定义的LanguageFiletype,IDE将相应类型的文件分析为由Psielement组成的树。示意图如图所示:
image.png

值得一提的是,后续Formatter、CompletionContributor等组件的实现受上述过程的影响很大,如果实现不好,必然会面临返工。但是有很多坑需要一个一个流过。这部分仅限于空间,不能写得太细。有兴趣看看Fortran的BNF定义,语言特征相对简单。

语法树单向对应
考虑到IDE内置对Java和Kotlin语言的支持,以及上一步的结果,我们得到了两棵语法树,是时候连接两棵树的节点了:

image.png

在这里,我们借用PsireferenceContributor(官方文档)注册Crelement(即Coderef语言Psielement的基类)引用源代码Psielement,基于每行双引号内容(字符串)。如何找到每个字符串对应的元素?遵循以下三个步骤:

除根节点外,每个节点还需要向上递归,找到每个级别的parent,直到根节点;

根节点是给定full-qualified-name的package或class,上一步的结果可以确定元素在package或class中的位置;

通过JavaPsiFacade和一系列搜索方法确定源中对应的Psielement;

注意:Kotlin Plugin 提供一套针对 Java 的 “Light” PsiElement 实现,因此这里我们考虑 Java 即可。

生成文档文本
有了语法树的对应关系,可以生成用于预览的文本。这部分比较常规,要时刻注意读写环境,按照以下步骤实现:

为每个Coderef语法树根节点指向的源代码文件创建副本;

遍历CodeRef树中的每一个Ref或Location,创建或定位副本中的相应位置,并将源代码文件中的元素(修改后)复制到副本中;

导出副本字符串;

考虑到 IDE 中 PSI 和文件是实时映射的,为不影响原文件内容,必须在副本环境中进行语法树的增删改。

虽然这部分不难,但繁琐程度最高。一方面,由于需要深入细节,上述KotlinLightPSI不再适用,因此必须分别为Java和Kotlin编写和实现。另一方面,如何保证复制后的代码格式正确也是一个大问题,尤其是元素之间的注释。最后,文本内容的生成在不断的断点和调试周期中形而上学地完成。

到目前为止,基建层的任务——将CodeRef还原为代码段——全部完成。

IDEA Plugin
有了前面的基础,IDEAPlugin主要负责使方案的本地使用体验可用易用。具体来说,插件的功能分为两类:

丰富语言功能的CodeRef;

面向Markdown,改进编辑,阅读体验;

接下来分别从以上角度介绍。

语言优化
对于一门新语言来说,从体验层面来说,PSI的完成只是第一步,自动完成、关键词亮度、格式化等功能对可用性的影响也是决定性的。尤其是在Coderef语法下,指望用户不依靠提示手动输入正确的包名、类名、方法名无疑太硬核了。让我们选择一些有趣的开始。

代码补全

在IDEA中,大多数(不太复杂的)代码都是通过Pattern模式注册的。所谓Pattern相当于Filter,当前光标位置满足Pattern时,会触发相应的ComplternConContributor。

我们可以用PlatformPaterns的几种内置方法来描述Pattern。例如,一个Coderef代码:method(“helloworld”),它的PSI树长如下:

  • CrMethodRef // text: method("helloWorld")

    • CrStringLiteral // text: "helloWorld"

      • LeafPsiElement // text: helloWorld

Pattern 因此为:

val pattern = PlatformPatterns.psiElement()

.withParent(CrStringLiteral::class.java)

.withSuperParent(2, CrMethodRef::class.java)

对应每一个Pattern,我们需要实现一个CompletionProvider给出补充信息,比如一个固定返回关键字补充的Provider:

val keywords = setOf("package", "class", "lang")

class KeywordCompletionProvider : CompletionProvider() {

override fun addCompletions(
    parameters: CompletionParameters,
    context: ProcessingContext,
    result: CompletionResultSet
) {

    keywords.forEach { keyword ->

        if (result.prefixMatcher.prefixMatches(keyword)) {

            // 添加一个 LookupElementBuilder,可以指定简单的样式

            result.addElement(LookupElementBuilder.create(keyword).bold())

        }

    }

}

}
掌握上述技能,如class、package、method等关键字,甚至方法名和字段名的补充都很容易实现。

比较 trick 的是包名和带有包名的类名的补全,它们形如 a.b.c.DEF。不同的是,每次输入 '.' 都会触发一次补全,而且要求在字符串开头直接输入“DE”也能正确联想并补全。限于篇幅不展开介绍了,详见 com.intellij.codeInsight.completion.JavaClassNameCompletionContributor 的实现。

格式化

在格式化方面,IDEA并没有直接使用PSI或ASTNode,而是建立了基于两者的Block系统。所有缩进和间距的调整都是以Block为最小粒度进行的(有些复杂的语言拆得太细,可以很好的降低设计的复杂性,很棒)。

这里概念不多,列举如下:

ASTBlock:我们用现有的ASTNode树构建Block,所以继承这个基础;

Indent:控制每行的缩进;

Spacing:控制每个Block之间的间距策略(最小、最大空间、强制换行/不换行等)。

Wrap:单行长度过长的折行策略;

Alignment:自己在ParentBlock中的对齐方向;

实际敲击代码时,大部分时间都花在getSpacing方法上,写出来的效果类似于这样:

override fun getSpacing(child1: Block?, child2: Block): Spacing? {

/*...*/

return when {

    // between ',' and ref

    node1?.elementType == CodeRefElementTypes.COMMA && psi2 is CrRef ->

        Spacing.createSpacing(/*minSpaces*/0, /*maxSpaces*/0, /*minLineFeeds*/1, /*keepLineBreaks*/true, /*keepBlankLines*/1)

    // between '[', literal, ']'

    node1?.elementType == CodeRefElementTypes.L_BRACKET && psi2 is CrStringLiteral ||

            psi1 is CrStringLiteral && node2?.elementType == CodeRefElementTypes.R_BRACKET ->

        Spacing.createSpacing(/*minSpaces*/0, /*maxSpaces*/0, /*minLineFeeds*/0, /*keepLineBreaks*/false, /*keepBlankLines*/0)

}

}
格式化属于说起来很简单,实现起来很头痛的东西。实操过程中,被迫把前面写好的 BNF 做了一波不小的调整,才达到理想效果。好在我们的语言比较简陋简洁,没踩到什么大坑,如果面向更复杂的语言,工作量将是指数级提升(参考 com.intellij.psi.formatter.java 包下的代码量)。

MarkdownX
上面列出了这么多内容,说白了就是Markdown中代码块的增强方案,然后Coderef和Markdown终于要合体了。

事实上,官方一直支持Markdown(IDEA内置,AS可选安装),包括一套完整的语言实现和编辑器,预览器。以下是预览的生成过程,如图所示:

image.png

分为以下步骤:

利用MarkdownParser将文本分析为多个ASTNode;

利用HtmlGenerator内置的visitor访问每个ASTNode生成HTML文本;

将生成的HTMLDocument设置为内置浏览器(如有),最终呈现在屏幕上;

交代个背景:在本项目启动之初,IDEA 正处于 JavaFX-WebView 到 JCEF 的过渡期(直接导致了 AndroidStudio 4.0 左右的版本没有可用的内置 WebView 实现)。

上述方案总结有以下问题:

兼容性差,部分IDE版看不到预览;

每一次MD变更都会触发完整的generateHtml,如果文档内容复杂度高,就会出现性能瓶颈;

将HTML文本set交给浏览器时,没有diff逻辑,会触发页面reload,这也可能导致性能问题(后来diff能力增加到带JCEF的IDE,但并非所有IDE都内置JCEF);

综合考虑,我们决定不直接使用本地插件,而是基于其创建新语言MarkdownX,最大限度地重用原有能力,增加对Coderef的支持,并根据Swing制作一套类似Recyclerview的机制来提高预览性能。

优化后的方案流程类似:

image.png

自制方案有许多优点:

内存占用较低(浏览器vs.JComponent)

性能更好(局部刷新、控件复用等)

更好的体验(浏览器内置对标签的支持太基础,无法实现代码亮度、引用跳转等功能,本地控件没有这些限制)

更好的兼容性(不解释)

CodeRef 支持

Markdownx只表现为新语言,MarkdownParser和HtmlGenerator在实现中仍然被重用,主要区别在于文件扩展名和code-fence的处理。

所谓code-fence,就是用``符号包裹在Markdown中的代码块。与本地实现不同,我们需要在生成预览时更换代码块的内容,并使内容随代码的变化而变化。

实际上,我们需要实现一个org.intellij.markdown.html.generatingProvider,简写如下:

class MarkDownXCodeFenceGeneratingProvider : GeneratingProvider {

override fun processNode(visitor: HtmlGenerator.HtmlGeneratingVisitor, text: String, node: ASTNode) {

    visitor.consumeHtml("<pre>")

    var state = 0 // 用于后面遍历 children 的时候暂存状态

    /* ...一些变量定义 */

    for(child in childrenToConsider) {

        if (state == 1 && child.type in listOf(MarkdownTokenTypes.CODE_FENCE_CONTENT, MarkdownTokenTypes.EOL)) {

            /* ...拼接每行内容 */

        }

        if (state == 0 && child.type == MarkdownTokenTypes.FENCE_LANG) {

            /* ...记录当前 code-fence 的语言 */

            applicablePlugin = firstApplicablePlugin(language) // 找到可以处理当前语言的“插件”

        }

        if (state == 0 && child.type == MarkdownTokenTypes.EOL) {

            /* ...进入代码段,设置状态 */

            state = 1

        }

    }

    if (state == 1) {

        visitor.consumeTagOpen(node, "code", *attributes.toTypedArray())

        if (language != null && applicablePlugin != null) {

            /* ...命中自定义处理逻辑(即 CodeRef)*/

            visitor.consumeHtml(content) // 即由自定义逻辑生成的 Html

        } else {

            visitor.consumeHtml(codeFenceContent) // 默认内容

        }

    }

    /* ...一些收尾 */

}

}
可此可见,当前代码段的语言可以在遍历node的children后确定。若语言为coderef,则进入上述预览文本生成逻辑,最后通过visitor(相当于HTMLBuilder)将定制内容拼接到Html中。

预览性能优化

考虑到JList没有item回收的能力,我们选择在List实现中直接使用Box。处理过程如下:

image.png

机制分为两大步:

Data层将HTML的body分成几个部分,diff后通知View层更改;

View层将变更的数据设置到List对应的位置,并尽可能重用现有的ViewHolder。该过程可能涉及创建和删除ViewHolder;

目前,我们为文本、图片和代码创建了三种ViewHolder:

文本:使用JTextPane与HTML+CSS一起恢复文本样式;

图片:自定义JComponent缩放,绘制,确保图片居中并完整显示;

代码:基于IDE提供的Editor,进行必要的设置和逻辑简化;

在这里,处理Editor花费了大量精力:

使用原始代码文件作为context创建PsicodeFragment作为内容填充Editor,以确保代码中的原始文件import类别、方法、字段可以正常resolve(这一点非常重要,如果使用MockDocument作为内容,绝大多数代码亮度和跳转不生效);

设置合适的HighlightingFilter,确保不报红(将原文件作为context的代价是当前代码片段的类很可能被认为是类重复,代码结构不一定合法,因此需要禁用报红级别的代码分析);

禁用Intention,设置只读(提高性能,减少干扰);

禁止Inspection和ExternalAnnotator;(两者都是性能消耗大户,后者包括AndroidLint相关逻辑)

经过以上优化,测量预览在大多数情况下可以顺利显示和刷新。但如果同时打开多个文档,或者操作速度惊人,还是会时不时卡住很久。分析发现,性能消耗主要在HTML生成上。

由于Markdown语法限制(节点深度低),传统MD转HTML的性能支出有限。但是回顾以上,我们对coderef的处理会伴随着大量的PSIresolve,复杂飙升,频繁的全generate就没那么合适了。一个很自然的想法是给每个coderef添加缓存,内容不变的时候直接使用缓存内容。这样,在修改文本段落时,可以完全避免其他文件的语法分析,在修改coderef段落时,只会刷新当前代码块的内容。

然后问题来了:如果用户修改的代码不是文档文件,而是被引用的代码,预览不会在缓存的作用下立即改变。那么进一步,如果你注册并监控所有被引用的文件,并在更改时刷新缓存,问题能解决吗?其实这样做的问题确实解决了,但是引入了新的问题:如何释放文件监控?

此处插入背景:对 code-fence 内容的干预是基于 Visitor 模式回调完成的,因此作为 generator 本身是不知道本次处理的代码块与前一次、后一次回调是否由同一个变更引起。举个例子:一个文档中有 A、B、C 三个 codeRef 块,则在一次 HTML 生成过程中,generator 会收到三次回调,且没有任何手段可以得知这三次回调的关联性。

目前,我们只能在HTML生成前后通知generator,并在generator内部维护一个队列+计数器,以不那么优雅地解决泄漏问题。

至此,插件的整体性能终于在可接受范围内。

Gradle / Dokka Plugin
为了让受众更广、内容随时可读,把文档做到可导出、可自动化部署是非常必要的。方案上,我们选用同为 IntelliJ 出品的 Dokka 作为基础框架,利用其完善的数据流变换能力,高效地适配多输出格式的场景。

Dokka 流程扩展

Dokka 作为同时兼容 Kotlin 和 Java 的文档框架,“数据流水线”的思想和极强的可扩展性是其特点。代码转换到文档页面的流程如下:

image.png

每个节点都有至少一个 Extension Point,扩展起来非常灵活。

图中几个主要角色列如下:

Env:包含基于KotlinCompiler和IntelliJ-Core扩展的代码分析器(用于输出DocumentModels)、开发者定制的插件等组件;

DocumentModels:对module、package、class、function、fields等元素的抽象,呈树形组织,本质上是一些dataclass;

PageModels:PageCreator以DocumentModels为输入,创建的一系列对象是封装页面,描述页面的结构;

Renderer:用于将PageModels渲染成某种格式的产品(Dokka内置HTML、Markdown等););

从以上内容可以看出,Doka最初的功能只是将代码转换为文档页面,而不是本地支持文档文件的转换(真的没有必要)。但是在我们的场景中,MarkdownX的渲染取决于源代码信息,所以Doka的这部分ka的这部分能力。

通过重写PageCreator,我们将包含Markdownx文档的项目变成这样一个节点树:
image.png

MdxDirNode 对应文件夹节点,页面内容是当前文件夹的目录,点击链接可跳转至下一级;

MdxPageNode 对应 MarkdownX 文档内容,包含若干类型的 children 分别代表不同类型的内容片段;

在创建MdxPagenode时,我们使用类似于上述IDEA-plugin的方法,重写一个org.jetbrains.doka.barsers.parserser,修改code-fence的处理,改为调用到基础设施部分生成coderef预览文本的代码,最终获得所需的文档文本。

飞书适配
在获得页面内容后,结合Doka自带的HTMLRenderer,很容易输出可用于部署的HTML产品。但目前的情况是,我们更喜欢将文档收敛到飞行书籍中,这需要为飞行书籍编写另一份定制Renderer。

考虑到自己处理页面的树形结构过于复杂,我们实际上是基于内置的Defaultrender基类来扩展的:

abstract class DefaultRenderer(

protected val context: DokkaContext

) : Renderer {

abstract fun T.buildHeader(level: Int, node: ContentHeader, content: T.() -> Unit)

abstract fun T.buildLink(address: String, content: T.() -> Unit)

abstract fun T.buildList(

    node: ContentList,

    pageContext: ContentPage,

    sourceSetRestriction: Set<DisplaySourceSet>? = null

)

abstract fun T.buildNewLine()

abstract fun T.buildResource(node: ContentEmbeddedResource, pageContext: ContentPage)

abstract fun T.buildTable(

    node: ContentTable,

    pageContext: ContentPage,

    sourceSetRestriction: Set<DisplaySourceSet>? = null

)

abstract fun T.buildText(textNode: ContentText)

abstract fun T.buildNavigation(page: PageNode)



abstract fun buildPage(page: ContentPage, content: (T, ContentPage) -> Unit): String

abstract fun buildError(node: ContentNode)

}
上面只列出一部分了回调方法。

可以看出,这种接口方式比较新颖:以Visitor的方式遍历页面节点树,然后为开发者提供一系列Builder/DSL风格的待实现方法。对于这些abstractfunction,内置Htmlrender采用kotlinx.html(DSL风格的HTML构建器)实现,这意味着我们也应该实现一套DSL风格的飞行文档构建器。

飞书开放平台文档查看链接:飞书开放平台

DSL部分不详细,这里主要讲飞书的文档结构。众所周知,Markdown在设计之初就是面向Web的,所以天生就有与HTML互动的能力。但飞书文档的数据结构相对更像Pdf、Docx等文件,层次有限,相对扁平。例如,在相同的文档内容中,MdxPagenode的结构是这样的:

image.png

而飞书的结构长这样:

image.png

可以看出,差异是巨大的。这部分差异的抹平完全取决于自定义的Feishurenderer,具体做法只能由casebycase介绍,仅限于篇幅不展开,一般思路是对不兼容的节点进行展开或合并,穿插必要的子树遍历。

以下是两个特殊点:图片和链接。

文档链接

写Markdown文档时,往往需要插入链接,指向其他Markdown文档(一般使用相对路径)。这时候就需要想办法把相对路径映射成飞书链接,需要在Render步骤之后进行,因为映射的时候需要知道相应文档的飞书链接是什么。

第一反应必须是对文档进行拓扑排序,并根据依赖关系逐一上传文档。但这需要文档之间没有循环依赖,这显然不能保证(两个文档相互引用相当常见)。幸运的是,飞行文档提供了修改文档的界面,因此我们可以提前创建一批空文档,在获得空文档的链接后更换相对路径。换句话说,文档上传的处理过程是:创建空文档->替换相对路径为相应的文档链接->修改文档内容。

图片

图片可以在Markdown中与文本并列,属于Paragraph的一种。在飞行图书文档结构中,图片属于Gallery,只能独占一行,不能与文本同行。这两种格式在实现中不能完全兼容。目前的初步实现方案是在Paragraph的Group入口处向下DFS,找到所有图片,提出放在文本前面。效果,只能忍受。

顺便说一下,图片也需要上传和替换逻辑,类似于文档链接,不重复。

结语

以上是文档套件的全部内容:基于InteliJ技术栈,我们通过设计新语言、编写IDE插件、Gradle/Dokka插件,形成了完整的文档辅助解决方案,有效建立了文档与代码的关联,大大提高了编写和阅读体验。

未来,我们将为框架引入更实用的改进,包括:

添加图形代码元素选择器,降低语言学习和使用成本;

优化预览渲染效果,对齐WebView;

探索部分框架(Dagger、Retrofit等)的文档自动生成能力。

目前框架还处于内部测试阶段,正在逐步扩大推广范围。方案成熟,功能稳定后,将整体开源方案,为更多用户服务,吸收社区Idea。请期待!

目录
相关文章
|
8月前
|
Python
空间管理大师已上线!(2),Python高级工程师进阶学习】
空间管理大师已上线!(2),Python高级工程师进阶学习】
|
Java p3c 开发者
阿里java开发规范学习(附P3C IDEA插件 帮助规范的养成)
浅析 阿里巴巴 Java 开发规约 (未完成) contents 为什么要学 编程规约 P3C IDEA 插件 why-use 我们知道,一般稍微大一点的公司,都会在系统架构设计完成之后,编码工作开始之前,给出一份属于自家公司,或是自家团队给出的编码规范文...
5340 0
|
8月前
|
开发者
一键自动化博客发布工具,用过的人都说好(阿里云篇)
使用一键自动化博客发布工具blog-auto-publishing-tools把博客发布到阿里云上。
一键自动化博客发布工具,用过的人都说好(阿里云篇)
|
8月前
|
JSON 开发者 数据格式
揭秘5.3k⭐开发者的秘密武器:it-tools在线工具集,你不可不知!
揭秘5.3k⭐开发者的秘密武器:it-tools在线工具集,你不可不知!
69 0
|
人工智能 自然语言处理 Java
当代码遇见AI:IDEA开启ChatGPT插件,分分钟成为高效程序猿!
当代码遇见AI:IntelliJ IDEA开启ChatGPT插件,分分钟成为高效程序猿!
2399 0
|
SQL IDE Linux
IDEA:7个强大功能助你高效编码和优质工作!
IDEA:7个强大功能助你高效编码和优质工作!
74 0
|
存储 Java 测试技术
【高效编码】关于IDEA调试的点点滴滴都在此文了。领导看了都说好!!!!
您好,我是码农飞哥,感谢您阅读本文!如果此文对您有所帮助,请毫不犹豫的一键三连吧。小伙伴们,有啥想看的,想问的,欢迎积极留言告诉我喔。 前面我写了一篇低效编码的文章,我很痛心。
176 0
【高效编码】关于IDEA调试的点点滴滴都在此文了。领导看了都说好!!!!
《蚂蚁金服高级开发工程师萧恺:IDEA 插件开发入门教程》电子版地址
蚂蚁金服高级开发工程师萧恺:IDEA 插件开发入门教程
81 0
《蚂蚁金服高级开发工程师萧恺:IDEA 插件开发入门教程》电子版地址
|
消息中间件 前端开发 Java
开发中遇到的问题&解决方案(十二)
由于之前做过贷款平台和电商平台,所以对于订单这个东西十分的敏感,有段时间有点疯狂的喜欢逮着京东、淘宝、拼多多的订单页看,思考别人在做的购物车和订单这块是怎么实现的,尝试找找Bug什么的,后面出去面试别人看我的项目也会问一些关于订单如何设计和实现的问题,所以感觉这个东西还是有讲的必要,下面进入主题。
253 0
开发中遇到的问题&解决方案(十二)
|
JSON 运维 前端开发
开发中遇到的问题&解决方案(十一)
前天不是开工嘛,然后刚刚到公司前端说测试环境好像挂了,开工就直接王炸了,找了运维,运维说服务器过年关机了回来发现有个配件坏了,暂时修不好。那我就本地部署一套当测试环境用,我同步了一份生产库到本地,然后问题就来了,之前好好的功能全部出现了问题,因为年前有需求改动,debug了好几遍代码也没有查出问题,然后突然想到MySQL版本不对。
152 0
开发中遇到的问题&解决方案(十一)