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

sxkk20081年前知识分享116

前言

前面的文章中,我们配置好了编辑器,实现了 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算法盒子:探索人工智能技术的未来发展

  人工智能(Artificial Intelligence,简称AI)作为当下热门的科技领域,已经取得了令人瞩目的成就。随着技术的不断进步,AI算法盒子正成为研究和应用...

ASR平台:革新性技术赋能语音识别领域的创新

ASR平台:革新性技术赋能语音识别领域的创新

  语音识别技术由于其广泛应用的需求不断增长,ASR平台在此背景下应运而生。ASR(Automatic Speech Recognition,自动语音识别)平台是一种基于...

生活黑科技让你的生活更舒适便捷,无需等待和烦恼

生活黑科技让你的生活更舒适便捷,无需等待和烦恼

  1. 智能家居控制系统  随着5G技术的普及,智能家居控制系统正日渐流行。通过智能连接,我们可以使用手机控制室内温度、照明、电视等各种家居设备,即使我们不再家也能灵活掌控...

人脸关键点:从识别技术到应用推广的发展

人脸关键点:从识别技术到应用推广的发展

  随着人工智能技术的不断发展,人脸关键点识别技术逐渐成为热门研究领域。人脸关键点识别技术是指通过计算机视觉和图像处理技术,在图像中准确地检测和定位人脸上的关键点,如眼睛、鼻...

高考在即,加油吧,少年—考前如何吃好睡好全力备战?

高考在即,加油吧,少年—考前如何吃好睡好全力备战?

  2023年高考全国统考于6月7日开端举行,详细科目考试时间安排为:  6月7日   9:00-11:30 语文  15:00-17:00 数学  6月8日   9:00-...

GPT-3.5 Turbo:下一代人工智能的未来

GPT-3.5 Turbo:下一代人工智能的未来

  人工智能(AI)的发展是许多科学家和工程师一直在关注和研究的话题。毫无疑问,AI已经在很多领域取得了巨大的进展,例如自然语言处理、语音识别、图像识别等领域。然而,随着技术...

发表评论    

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