实现一个 Code Pen:(四)浏览器编译代码

sxkk20082年前知识分享195

前言

前面的文章中,我们配置好了编辑器,实现了 css、html、js 的编辑,现在我们需要做代码实时运行的功能了,并且可以直接写 less、scss、可以写 JavaScript、typescript、react。这个就涉及到了浏览器编译代码的逻辑,前期我们编译语言少一点、先把整体流程跑通,后面可以对语言和功能再慢慢丰富,这也是做项目的主要思路。

Iframe 实时运行

想要一个页面实时运行,并且 JS 变量不污染全局,Iframe 是一个不错的选择,得益于 iframe 有一个 srcDoc,我们可以直接更改里面的内容,页面就会实时变更和渲染, 业内的编辑器也是这么做的,一起看看下最简单的实现代码吧。

import React, { useState, useEffect } from 'react'
import Editor from './Editor'
import { useLocalStorage } from 'react-use'

function App() {
  const [html, setHtml] = useLocalStorage('html', '')
  const [css, setCss] = useLocalStorage('css', '')
  const [js, setJs] = useLocalStorage('js', '')
  const [srcDoc, setSrcDoc] = useState('')

  useEffect(() => {
    const timeout = setTimeout(() => {
      setSrcDoc(`
        
          
            
          
          ${html}
          
        
      `)
    }, 800)

    return () => clearTimeout(timeout)
  }, [html, css, js])

  return (
    <>
      <div className="pane">
        <Editor language="html" value={html} onChange={setHtml} />
        <Editor language="css" value={css} onChange={setCss} />
        <Editor language="javascript" value={js} onChange={setJs} />
      </div>
      <div className="pane">
        <iframe
          srcDoc={srcDoc}
          title="output"
          sandbox="allow-scripts"
          frameBorder="0"
          width="100%"
          height="100%"
        />
      </div>
    </>
  )
}

export default App

首先我们安装了react-use, 这个 hooks 是目前比较流行的 hook 库,使用useLocalStorage, 将数据存储到 LocalStorage 中,这样可以放在刷新页面的时候数据丢失。当然这是最简单的代码逻辑,为了防止整个 iframe dom 的销毁和重建,我使用 postMessage,具体代码可以直接看 Github

JS 编译

以上代码逻辑, 编辑器实现了原生 js 和 css 的支持,但是不支持 react 和 typescript,若要支持,需要在插入 srcDoc 之前将代码表编译成 es5,其实 babel 有个游览器版本@babel/standalone,并且有 presets 预设,支持 react 和 typescript, 只需要引入 srcipt 就可以,详情可以参考官方文档

<script src="https://unpkg.com/@babel/standalone/babel.min.js">script>
<script type="text/babel" data-presets="typescript">
  const x: number = 0
  console.log(x)
script>

以上代码就可以支持在浏览器执行

接下来我们需要支持 react 代码

import * as Babel from '@babel/standalone'

function compileJs(code) {
  const res = Babel.transform(code, {
    presets: ['react'],
  })
  return res.code
}

其实也很简单只需要设置 presets 设置为 react 就可以将编译 jsx 为 es5 了。

编译 typescript

编译 typescript 也是如此,需要注意的是 typescript 需要传入一个 filename 才可以

function compileTs(code) {
  const res = Babel.transform(code, {
    presets: ['typescript'],
    filename: 'index.ts',
  })
  return res.code
}

Less 编译

大部分同学都知道 less 使用的 2 种方式

  1. 在 Node.js 环境中使用 Less
npm install -g less
lessc styles.less styles.css
  1. 在浏览器环境中使用 Less
<link rel="stylesheet/less" type="text/css" href="styles.less" />
<script src="//cdnjs.cloudflare.com/ajax/libs/less.js/3.11.1/less.min.js">script>
  1. 我们的需求也是在浏览器中执行,但我们可以将编译的逻辑放在 web worker 中
import Less from 'less/lib/less'
const less = Less()
less.PluginLoader = function () {}

async function compileLess(code) {
  return await less.render(code).then((res) => res.css)
}

Scss 编译

scss 编译我选择的是 sass.js

同样首先需要安装

npm install -g sass.js

安装完成后,可以看下 node_modules 中的目录

sass.js 目录

我们发现目录中有个 sass.worker.js, 这个就 编译的 web worker js 代码, sass.js 已经将编译的逻辑独立到了这个 js 中,使用的时候需要设置 worker 的路径。 所以我们需要手动拷贝 node_modules 下的 sass.worker.jspublic/vendor 中,下面是实现代码

import Sass from 'sass.js/dist/sass'
Sass.setWorkerUrl('/vendor/sass.worker.js')

function compileScss(code) {
  const sass = new Sass()
  return new Promise((resolve, reject) => {
    sass.compile(code, (result) => {
      if (result.status === 0) return resolve(result.text)
      reject(new Error(result.formatted))
    })
  })
}

小结

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

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

本篇中浏览器编译的代码都很简单,但我却花了我几天时间,主要是这些代码都用的比较少,我又需要将编译的逻辑放入 web worker 中,然而 web worker 又没有 document 对象,所以不能直接使用 browser 版本的 js。当然目前还没实现 react typescript 的编译功能,先不卡在这了,把这项功能加入到 Todo List 中吧。

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

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

相关文章

数字经济:如何推动经济创新与发展

数字经济:如何推动经济创新与发展

  在当今全球化和科技高速发展的时代,数字经济作为引领经济发展的重要力量,正日益成为各国政府和企业的关注焦点。数字经济以信息和通信技术为基础,借助互联网、大数据、人工智能等先...

阿里AI人工智能平台:引领智能时代的创新驱动力

阿里AI人工智能平台:引领智能时代的创新驱动力

  在当今数字化时代,人工智能正成为推动企业创新和发展的关键力量。阿里巴巴作为全球领先的科技巨头,旗下的AI人工智能平台为企业提供了一系列创新技术和解决方案,帮助他们实现数字...

什么是供应链金融?

什么是供应链金融?

  供应链金融是一种结合金融与供应链管理的业务模式,它通过对供应链上下游企业的财务与物流数据进行整合和分析,为供应链参与者提供融资、结算、风险管理等金融服务,促进供应链各环节...

AI技术安全-防范风险与保障隐私的挑战与机遇

AI技术安全-防范风险与保障隐私的挑战与机遇

  现如今,人工智能(AI)技术正以惊人的速度不断发展,构建着各行各业的数字世界。与此同时,随着数字经济、智能化生产、信息化社会等趋势的不断加速,AI技术的应用也在日益扩大。...

AI绘画生成器:你亲手创作的艺术品

AI绘画生成器:你亲手创作的艺术品

  近年来,AI技术的不断发展,让人工智能在艺术领域得以应用,尤其是AI绘画生成器网站,通过深度学习、神经网络等先进技术,将传统艺术与现代科技相结合,为广大用户提供了一个创作...

AI技术在音乐创作中的应用与发展

AI技术在音乐创作中的应用与发展

  近年来,人工智能技术的发展已经悄然影响到了传统音乐创作领域。人工智能音乐生成技术,也被称为AI作曲技术。此类技术不仅受到专业音乐创作者的欢迎,同时也受到音乐爱好者的追逐和...

发表评论    

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