• 跳转到 … +
    browser.coffee cake.coffee coffeescript.coffee command.coffee grammar.coffee helpers.coffee index.coffee lexer.coffee nodes.coffee optparse.coffee register.coffee repl.coffee rewriter.coffee scope.litcoffee sourcemap.litcoffee
  • browser.coffee

  • §

    这个 **浏览器** 兼容层扩展了核心 CoffeeScript 函数,以便在浏览器中直接编译代码时能够顺利运行。我们添加了对通过 **XHR** 加载远程 Coffee 脚本、text/coffeescript 脚本标签、通过 data-URL 加载源映射等的支持。

    CoffeeScript = require './coffeescript'
    { compile } = CoffeeScript
  • §

    使用 window.eval 来评估代码,而不是仅仅使用 eval,以便在干净的全局作用域中运行脚本,而不是继承 CoffeeScript 编译器的作用域。(因此 cake test:browser 也能在 Node 中运行,请根据需要使用 window.eval 或 global.eval)。

    CoffeeScript.eval = (code, options = {}) ->
      options.bare ?= on
      globalRoot = if window? then window else global
      globalRoot['eval'] compile code, options
  • §

    运行代码无法访问此作用域。

    CoffeeScript.run = (code, options = {}) ->
      options.bare      = on
      options.shiftLine = on
      Function(compile code, options)()
  • §

    导出比 index.coffee 导出的更有限的 CoffeeScript,后者是为 Node 环境设计的。

    module.exports = CoffeeScript
  • §

    如果我们不在浏览器环境中,我们就完成了公共 API 的工作。

    return unless window?
  • §

    尽可能包含源映射。如果我们有 base64 编码器、JSON 序列化器和用于转义 Unicode 字符的工具,我们就可以开始了。从 https://mdn.org.cn/en-US/docs/DOM/window.btoa 移植而来

    if btoa? and JSON?
      compile = (code, options = {}) ->
        options.inlineMap = true
        CoffeeScript.compile code, options
  • §

    通过 XHR 从当前域加载远程脚本。

    CoffeeScript.load = (url, callback, options = {}, hold = false) ->
      options.sourceFiles = [url]
      xhr = if window.ActiveXObject
        new window.ActiveXObject('Microsoft.XMLHTTP')
      else
        new window.XMLHttpRequest()
      xhr.open 'GET', url, true
      xhr.overrideMimeType 'text/plain' if 'overrideMimeType' of xhr
      xhr.onreadystatechange = ->
        if xhr.readyState is 4
          if xhr.status in [0, 200]
            param = [xhr.responseText, options]
            CoffeeScript.run param... unless hold
          else
            throw new Error "Could not load #{url}"
          callback param if callback
      xhr.send null
  • §

    通过让浏览器编译和评估所有内容类型为 text/coffeescript 的脚本标签来激活浏览器中的 CoffeeScript。这在页面加载时发生。

    CoffeeScript.runScripts = ->
      scripts = window.document.getElementsByTagName 'script'
      coffeetypes = ['text/coffeescript', 'text/literate-coffeescript']
      coffees = (s for s in scripts when s.type in coffeetypes)
      index = 0
    
      execute = ->
        param = coffees[index]
        if param instanceof Array
          CoffeeScript.run param...
          index++
          execute()
    
      for script, i in coffees
        do (script, i) ->
          options = literate: script.type is coffeetypes[1]
          source = script.src or script.getAttribute('data-src')
          if source
            options.filename = source
            CoffeeScript.load source,
              (param) ->
                coffees[i] = param
                execute()
              options
              true
          else
  • §

    options.filename 定义了源映射在开发者工具中显示的文件名。如果脚本标签有 id,则使用它作为文件名;否则使用 coffeescript 或 coffeescript1 等,将第一个保留为未编号,以应对只有一个 CoffeeScript 脚本块要解析的常见情况。

            options.filename = if script.id and script.id isnt '' then script.id else "coffeescript#{if i isnt 0 then i else ''}"
            options.sourceFiles = ['embedded']
            coffees[i] = [script.innerHTML, options]
    
      execute()
  • §

    监听窗口加载事件,无论是在正常的浏览器还是 IE 中。仅在启动时为非 ES 模块版本的浏览器编译器附加此事件处理程序,以保持向后兼容性,同时允许 ES 模块版本在没有副作用的情况下导入。

    if this is window
      if window.addEventListener
        window.addEventListener 'DOMContentLoaded', CoffeeScript.runScripts, no
      else
        window.attachEvent 'onload', CoffeeScript.runScripts