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

sxkk20081年前知识分享79

在上文中,我们使用 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 */ `
`'
))
// Output: const html = /* HTML */ `
`;
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 的标准,因此我们可以覆盖原先的格式化指令, 主要通过monaco.languages.registerDocumentFormattingEditProvider 来实现。

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 优化格式化代码的性能;

关于 Monaco Editor 的配置请参考官网Github

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

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

本文首发掘金平台,来源Ai知识分享博客

相关文章

电影人工智能:开启新时代的创作与体验之旅

电影人工智能:开启新时代的创作与体验之旅

  随着科技的不断进步和人工智能的快速发展,电影产业正逐渐步入一个崭新的时代。人工智能在电影创作、制作、展示和观赏等方面的应用逐渐成熟,为电影行业带来了前所未有的变革和机遇。...

AI手绘板绘画:数字科技与创意艺术的完美结合

AI手绘板绘画:数字科技与创意艺术的完美结合

  AI手绘板绘画技术在数字时代的发展与普及中逐渐走进人们的视野与生活。它不仅为广大绘画爱好者和设计师提供了更加便捷、高效、精准的创作手段,更让数字科技与创意艺术得以完美结合...

AI与未来——探索人工智能发展趋势与影响

AI与未来——探索人工智能发展趋势与影响

  人工智能(AI)作为当下最热门的技术领域,正在改变人类的生产方式、生活方式和认知方式。AI的发展速度迅猛,在医疗、金融、教育等各行各业中都有着广泛应用。未来,AI将会给我...

AI识物机器人—让生活更智能化

AI识物机器人—让生活更智能化

  随着科技的不断革新,人工智能技术正在逐渐渗透到我们的生活中。其中一项应用便是AI(人工智能)识物机器人,它的出现不仅仅是一种科技的突破,更是推动了智能化生活的革命,成为了...

数字经济是什么?数字经济的概念、特征和影响

数字经济是什么?数字经济的概念、特征和影响

  数字经济是指以数字技术为基础,利用互联网、大数据、人工智能等技术手段进行经济活动的一种新型经济形态。在数字经济时代,信息技术的快速发展改变了人们的生产、生活方式,对经济发...

Compressor.io 

Compressor.io 可以使用有损或无损压缩优化 JPEG、PNG、SVG、GIF 和 WebP,每个文件最高可达 10MB。如果要自定义压缩或使用较大的文件,则必须付费获得高级计划。在这种情况下,压缩图片在服务器端完成,因此将获得更快的结果。

Compressor.io

Imagecompresser.com 

Imagecompresser.com 允许同时上传多达 10 个文件,并支持 PNG、JPEG、WebP、JPG 和 GIF 格式。每个图像的文件大小没有任何限制,因此这可能适用于 10 个或更少的大文件。

Imagecompresser.com

30+ 图片压缩工具集合,包含在线压缩和CLI工具

许多开发人员花费了大量时间优化网页性能,比如优化 js、css、减少 http 请求等等,但减小图片大小产生的优化比其他所有领域加起来影响更大。WebUtils Bulk Image Compress...

发表评论    

◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。