实现一个 Code Pen:(三)10 行代码实现代码格式化

简介: 在上文中,我们使用 monaco-editor 结合 Next.js,打造了编辑器的功能,在本文中,我们将继续优化 monaco-editor, 使它拥有代码格式化的功能。

highlight: monokai

一起养成写作习惯!这是我参与「掘金日新计划 · 4 月更文挑战」的第 8 天,点击查看活动详情

在上文中,我们使用 monaco-editor 结合 Next.js,打造了编辑器的功能,在本文中,我们将继续优化 monaco-editor, 使它拥有代码格式化的功能。

prettier 在浏览器使用

关于代码格式化,被人熟悉的是 prettier,在前端工程中,为了保证团队成员提交代码的格式一致,会先安装 prettierhusky,使用 Git hooks 函数,在代码提交之前把代码格式化,此时的 prettier 是 nodejs 版本,是一个可执行的 cli 工具, 当然 prettier 也有 Browser 版本,也就是 prettier/standalone, 现代浏览器都支持 ES modules, 通过下面这几行代码就可以实现浏览器端代码格式化了。

<script type="module">
  import prettier from "https://unpkg.com/prettier@2.6.2/esm/standalone.mjs";
  import parserBabel from "https://unpkg.com/prettier@2.6.2/esm/parser-babel.mjs";
  import parserHtml from "https://unpkg.com/prettier@2.6.2/esm/parser-html.mjs";

  function formatCode(code){
    
    return prettier.format(code, {
    
      parser: "babel",
      plugins: [parserBabel, parserHtml],
    })
  }

  console.log(formatCode("const html=/* HTML */ `<DIV> </DIV>`"));
  // Output: const html = /* HTML */ `<div></div>`;
</script>

prettier 使用方法的核心就是调用不同的 parser,去解析不同的文本,在我当前的开发的 Code Pen 场景中,使用到了以下几个 parser:

  • babel: 处理 js
  • html: 处理 html
  • postcss: 用来处理 css, less, scss
  • typescript: 处理 ts

除了 ES modules 方式, Prettier 浏览器版本,还支持 amd, commonjs 的用法,使用非常方便。详情用法可以查看官方文档

集成到 monaco-editor

monaco-editor 本身也提供了格式化的命令,可以通过右键菜单或者快捷键⇧ + ⌥ + F来对代码进行格式化,目前自带的格式化工具不如 Prettier 的标准,因此我们可以覆盖原先的格式化指令, 主要通过来实现。

import * as monaco from "monaco-editor/esm/vs/editor/editor.api";

const modelToPaser={
   
  html
  css,
  less,
  sass,
  typescript,
  javascript:'babel'
}
export function registerDocumentFormattingEditProviders() {
   
  const disposables = [];

  const formattingEditProvider = {
   
    async provideDocumentFormattingEdits(model, _options, _token) {
   
      const pretty=formatCode( model.getValue(), modelToPaser[model.getLanguageId()])
      if (canceled || error) return [];
      return [
        {
   
          range: model.getFullModelRange(),
          text: pretty,
        },
      ];
    },
  };

 ['html','css','less','scss','javascript','typescript'].forEach((id)=>{
   
  disposables.push(
    monaco.languages.registerDocumentFormattingEditProvider(
      id,
      formattingEditProvider
    )
  );
 })

  return {
   
    dispose() {
   
      disposables.forEach((disposable) => disposable.dispose());
    },
  };
}

上述代码中 通过 model.getValue() 获得当前编辑器中的文本,通过 model.getLanguageId() 获得当前编辑器的编程语言,每一种语言都有不同的解析器,需要与Prettier的 paser 对应,比如:JavaScript 语言对应的就是babel paser。

至此,整个 Prettier 的流程便已完成,为了提高解析性能,可以将格式化的代码放入一个 web worker 中,完整的 web worker 代码如下:

import prettier from "prettier/standalone";

const options = {
   
  html: async () => ({
   
    parser: "html",
    plugins: [await import("prettier/parser-html")],
    printWidth: 100,
  }),
  typescript: async () => ({
   
    parser: "typescript",
    plugins: [await import("prettier/parser-typescript")],
    printWidth: 100,
  }),
  css: async () => ({
   
    parser: "css",
    plugins: [await import("prettier/parser-postcss")],
    printWidth: 100,
  }),
  less: async () => ({
   
    parser: "less",
    plugins: [await import("prettier/parser-postcss")],
    printWidth: 100,
  }),
  scss: async () => ({
   
    parser: "scss",
    plugins: [await import("prettier/parser-postcss")],
    printWidth: 100,
  }),
  javascript: async () => ({
   
    parser: "babel",
    plugins: [await import("prettier/parser-babel")],
    printWidth: 100,
    semi: false,
    singleQuote: true,
  }),
};

let current;

addEventListener("message", async (event) => {
   
  if (event.data._current) {
   
    current = event.data._current;
    return;
  }

  function respond(data) {
   
    setTimeout(() => {
   
      if (event.data._id === current) {
   
        postMessage({
    _id: event.data._id, ...data });
      } else {
   
        postMessage({
    _id: event.data._id, canceled: true });
      }
    }, 0);
  }

  const opts = await options[event.data.language]();

  try {
   
    respond({
   
      pretty: prettier.format(event.data.text, opts),
    });
  } catch (error) {
   
    respond({
    error });
  }
});

覆盖快捷键

相比于 cmd + s 时,执行自定义的函数,不如直接覆盖掉自带的格式化指令,在 cmd + s 时直接执行指令来完成格式化来的优雅。执行上面的代码就已经覆盖格式化的指令,接下来,只需要绑定快捷键就可以了。

function setupKeybindings(editor) {
   
  let formatCommandId = "editor.action.formatDocument";
  const {
    handler, when } = CommandsRegistry.getCommand(formatCommandId);
  editor._standaloneKeybindingService.addDynamicKeybinding(
    formatCommandId,
    monaco.KeyMod.CtrlCmd | monaco.KeyCode.KeyS,
    handler,
    when
  );
}

通过 CommandsRegistry.getCommand(formatCommandId) 获得 action 的方法,在通过 _standaloneKeybindingService.addDynamicKeybinding 绑定快捷键。然后在 react 组件初始化的时候绑定快捷键就可以了

useEffect(() => {
   
    if (divEl.current) {
   
      editor.current = monaco.editor.create(divEl.current, {
   
        minimap: {
    enabled: false },
        theme: "vs-dark",
      });
    }

    setupKeybindings(editor.current);

    return () => {
   
      editor.current.dispose();
    };
  }, []);

至此我们编辑器快捷键格式化的逻辑就完成了。

格式化效果

预览地址:https://code.runjs.cool/pen/1

代码仓库:https://github.com/maqi1520/next-code-pen

小结

  • 使用prettier/standalone在浏览器代码格式化;
  • monaco.languages.registerDocumentFormattingEditProvider 修改 monaco 默认的格式化代码方法;
  • editor._standaloneKeybindingService.addDynamicKeybinding 绑定快捷键;
  • 使用 web worker 优化格式化代码的性能;

接下来将介绍代码在线编译的实现。

以上就是本文全部内容,希望这篇文章对大家有所帮助,也可以参考我往期的文章或者在评论区交流你的想法和心得,欢迎一起探索前端。

本文首发掘金平台,来源小马博客

相关文章
|
前端开发 JavaScript API
MonacoEditor 加载很慢该怎么优化?
MonacoEditor 加载很慢该怎么优化?
2978 0
|
9月前
|
SQL 关系型数据库 MySQL
【亲测有用】数据集成平台能力演示(支持国产数据库DaMeng与KingBase)
杭州奥零数据科技有限公司成立于2023年,专注于数据中台业务,维护开源项目AllData并提供商业版解决方案。AllData提供数据集成、存储、开发、治理及BI展示等一站式服务,支持AI大模型应用,助力企业高效利用数据价值。
【亲测有用】数据集成平台能力演示(支持国产数据库DaMeng与KingBase)
|
人工智能 自然语言处理 数据可视化
深耕智能文档处理“百宝箱”,合合信息为文档研发注入新动力
在1024程序员节上,合合信息发布了智能文档处理“百宝箱”,包括可视化文档解析工具TextIn ParseX、向量化模型acge-embedding和文档解析测评工具markdown_tester,全面提升文档解析与管理的效率和准确性,广泛应用于知识库构建、智能文档抽取、大模型训练数据治理和文档翻译等多个领域。
解决ERROR in Conflict: Multiple assets emit different content to the same filename index.html 的问题
解决ERROR in Conflict: Multiple assets emit different content to the same filename index.html 的问题
1214 1
|
资源调度 JavaScript 索引
Vue2开发插件并发布到npm
这篇文章介绍了如何使用Vue 3、TypeScript和Vite开发一个下拉框组件`vue-amazing-selector`,并将其发布到npm,包括了项目的创建、组件开发、配置webpack、编写组件代码、导出组件、编译、npm包初始化、发布流程以及在项目中使用该插件的完整步骤。
286 0
Vue2开发插件并发布到npm
|
前端开发 数据可视化 项目管理
Dhtmlx Gantt教程:创建交互式甘特图的完整指南
Dhtmlx Gantt教程:创建交互式甘特图的完整指南
|
机器学习/深度学习 数据采集 人工智能
数据工作中的自动化与AI融合实践
【8月更文第13天】随着大数据和人工智能(AI)技术的发展,数据处理和分析变得越来越重要。本文将探讨如何通过自动化工具和AI技术来优化数据处理流程,包括数据清洗、特征工程、模型训练以及结果可视化等步骤。我们将使用Python编程语言及其相关库(如Pandas、Scikit-learn和TensorFlow)作为实现手段。
916 0
|
XML Android开发 数据格式
Error obtaining Ul hierarchy Reason: Error while obtaining Ul hierarchy XML file
Error obtaining Ul hierarchy Reason: Error while obtaining Ul hierarchy XML file
722 1
|
存储 JSON C#
Unity 数据读取|(四)Json文件解析(Newtonsoft.Json ,Litjson,JsonUtility,SimpleJSON)
Unity 数据读取|(四)Json文件解析(Newtonsoft.Json ,Litjson,JsonUtility,SimpleJSON)
|
Docker 容器
docker 设置国内镜像源
docker 设置国内镜像源
91476 1