mustache-4.1.0源码注解

由于1.0以后变化都不是很大,所有就直接看最新的源码解析了
https://cdn.bootcdn.net/ajax/libs/mustache.js/4.1.0/mustache.js

相对于以前 这个版本去掉了 !注释 和 < 获取新模板 渲染也去掉了

大概的方法

  • {{name}} ---- 会把数据中 name 转换为具体值
  • {{#list}} {{/list}} ----会把list中的数据选项展示
  • {{#flag}}flag为ture展示{{/flag}}{{^flag}}flag为false展示{{/flag}} ----判断语句
  • {{{}}} ---- 里面的内容不会转义
  • {{>name}} ----会从补充数据中获取数据(接口返回的数据不足以展示,需要补充数据)
  • <div>{{=<% %>=}}</div> ----修改 <% %> 为新切割符号

原理

大概原理是 把template模板 通过 对应的分隔符号 比如 {{ }} 转为为tokens;

["text", "</div>↵", 181, 188]
["^", "list", 194, 203, Array(5), 260]
["text", " <span>姓名:", 204, 221]
["name", "name", 221, 229]
["text", "<i>年龄:", 229, 235]
["name", "age", 235, 242]
["text", "</i></span>↵", 242, 254]
["/", "list", 260, 269]
["#", "arr", 276, 284, Array(3), 321]

[type, value, start, scanner.pos]

  • 其中type 是这个 类型 text 为文本, name需要取数据中的键 ,# 是循环
  • value 数据中的键或者字符串
  • start 这个是当前数组所在字符串的位置
  • scanner.pos 扫描器的位置或者未结束位置

然后把tokens处理为 dom识别的字符串;

栗子:

      var template = `
        <div>{{name}}</div>       

        {{#arr1}}
            <p>{{.}}</p>
        {{/arr1}}

      `;
      var data = {
        name: '张三',        
        arr1: ['第一个', '第二个']
      };
      const str = Mustache.render(template, data);
image.png

image.png

源码里面有三个类:

  • Scanner;
  • Context;
  • Writer;

Scanner 扫描器

我的理解 Scanner 是 当 匹配符号比如 {{ 和 }} 匹配一段文本是分别 获取 匹配符号之外的文本。(其实跟以前版本正则类似)
比例: 我是{{name}},我爱我的{{contry}}。也爱你
通过Scanner类可以获取到 五段内容:

  1. 我是
  2. name
  3. ,我爱我的
  4. contry
  5. 。也爱你

Context 上下文

作用是用来处理data数据,掉push方法, 可以把循环中的token 存入 当前token中。便于查找对应的数据

Writer

提供render方法。 把模板转换为tokens。然后处理只有的tokens 通过数据转换为浏览器识别的字符串

html 测试

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta http-equiv="X-UA-Compatible" content="IE=edge" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <title>Document</title>
  </head>
  <body>
    <div id="app"></div>

    <script src="./mustache4.0.js"></script>
    <script>
      // !注释 去掉了 
      // < 获取新模板 渲染也去掉了
      var template = `
      <div>{{name}}</div>
      
      <div>{{age}}</div>

      <div>{{!我是注释}}</div> 

      <div>{{<template2}}</div>

      <div>{{{f1}}}</div> 

      <div>{{{f2}}}</div> 

      {{#list}}list存在显示我{{/list}}
      {{^list}}list不存在显示<span>姓名:{{name}}<i>年龄:{{age}}</i></span>{{/list}}

      {{#arr}}
        <span>{{test}}</span>
      {{/arr}}

      {{#arr1}}
          <p>{{.}}</p>
      {{/arr1}}

      {{#arr2}}
          <p>{{top}}</p>
      {{/arr2}}

      {{#arr3}}
          <p>{{99}}</p>
      {{/arr3}}  

      <div>{{>abc}}</div>
      <div>{{=<% %>=}}</div>
      <div><%name%></div>
      `;
      var template2 = '<sapn>{{sex}}</sapn';
      var data = {
        name: '张三',
        age: 14,
        f1: 'abc&<',
        f2: function () {
          return 666;
        },
        f3: 'abc&<',
        sex: '男',
        test: 'tttttttt',
        list: [], //也可以不要这一项
        arr: () => (template, fn) => {
          return fn(template);
        },
        arr1: ['第一个', '第二个'],
        arr2: [{ top: 'test1' }, { top: 123 }],
        arr3: [{ 99: 'test1' }, { 99: 123 }],
      };
      const str = Mustache.render(template, data, { "abc": '123', bbb: 'ttt', ccc: 'ttt', ddd: 'ttt' });
      // 也可以传入函数
      //const str = Mustache.render(template, data, (a)=>{ return a + '333' });
      console.log('str', str);
      document.querySelector('#app').innerHTML = str;
    </script>
  </body>
</html>

源码

// umd 写法: 兼容AMD和commonJS规范的同时,还兼容全局引用的方式
(function (global, factory) {
  typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :
  typeof define === 'function' && define.amd ? define(factory) :
  (global = global || self, global.Mustache = factory());
}(this, (function () { 'use strict';



  // 定义 判断 数组方法
  var objectToString = Object.prototype.toString;
  var isArray = Array.isArray || function isArrayPolyfill (object) {
    return objectToString.call(object) === '[object Array]';
  };

  // 判断 函数方法
  function isFunction (object) {
    return typeof object === 'function';
  }

  // 获取 obj 的类型 
  function typeStr (obj) {
    return isArray(obj) ? 'array' : typeof obj;
  }

  // escapeRegExp是把匹配到的符号  前加一个反斜杠
  /**拓展: 
  * $1、$2、...、$99: 与 regexp 中的第 1 到第 99 个子表达式相匹配的文本
  * $&:与 regexp 相匹配的子串
  * $`:位于匹配子串左侧的文本。
  * $':位于匹配子串右侧的文本。
  * $$:直接量符号。
  * */
  function escapeRegExp (string) {
    return string.replace(/[\-\[\]{}()*+?.,\\\^$|#\s]/g, '\\$&');
  }

  /**
   * 检查对象 是否具有给定属性的方法
   */
  function hasProperty (obj, propName) {
    return obj != null && typeof obj === 'object' && (propName in obj);
  }

  /**
   * 检测一个属性是否是对象的自有属性 (前提这个 对象不是null 并且不是 object)js中万物皆对象。
   */
  function primitiveHasOwnProperty (primitive, propName) {
    return (
      primitive != null
      && typeof primitive !== 'object'
      && primitive.hasOwnProperty
      && primitive.hasOwnProperty(propName)
    );
  }


  var regExpTest = RegExp.prototype.test;
  // 正则 test 方法
  function testRegExp (re, string) {
    return regExpTest.call(re, string);
  }

  var nonSpaceRe = /\S/;
  // 正则判断是 空白(包括 空格、换行、tab缩进等空白)
  function isWhitespace (string) {
    return !testRegExp(nonSpaceRe, string);
  }

  // 默认 需要转义的  符号
  var entityMap = {
    '&': '&amp;',
    '<': '&lt;',
    '>': '&gt;',
    '"': '&quot;',
    "'": '&#39;',
    '/': '&#x2F;',
    '`': '&#x60;',
    '=': '&#x3D;'
  };

  // 转义 特殊符号
  function escapeHtml (string) {
    return String(string).replace(/[&<>"'`=\/]/g, function fromEntityMap (s) {
      return entityMap[s];
    });
  }

  var whiteRe = /\s*/; // 空格可能存在
  var spaceRe = /\s+/; // 最少有一个空格
  var equalsRe = /\s*=/; // 等于号前面可能存在空格
  var curlyRe = /\s*\}/; // 等于号前面最少存在一个空格
  var tagRe = /#|\^|\/|>|\{|&|=|!/; // 最少存在匹配里面一个符号

  // 处理模板为tokens
  function parseTemplate (template, tags) {
    // 如果没有传入模板,直接返回空数组
    if (!template)
      return [];
    var lineHasNonSpace = false; // 标志 当前行 是否有内容
    var sections = [];     // 栈(先进后出,后进先出)
    var tokens = [];       // 获取的tokens  (会切割得很细,一个空格也会是一个token)
    var spaces = [];       // 这个数据会保存 tokens 中 保存的是空格 的位置
    var hasTag = false;    // 当前行上有{{tag}}吗? 存在 解析标签 (stripSpace方法 用来去除空格用)
    var nonSpace = false;  // 当前行中是否有非空格字符? (stripSpace方法 用来去重空格用)
    var indentation = '';  // 跟踪使用它的标记的缩进  (记录有多少空格)
    var tagIndex = 0;      // 存储一行中遇到的标记数

    // 去除当前行的所有空白标记数组, 如果上面有{{#tag}} 并且 只有空格。
    function stripSpace () {
      if (hasTag && !nonSpace) {
        // 根据 保存的空格 位置, 删除tokens中的空格token
        while (spaces.length)
          delete tokens[spaces.pop()];
      } else {
        spaces = [];
      }
      // 把 匹配符号 标志重置为fals。 nonSpace为false
      hasTag = false;
      nonSpace = false;
    }


    var openingTagRe, closingTagRe, closingCurlyRe;
    // 根据传入的符号 设置标签
    function compileTags (tagsToCompile) {
      // 如果 匹配符号是 字符串, 说明要用自定义的解析符号 
      // 比如 <% %>  安装空字符串切个我一个对象 ["<%", ">%"]
      if (typeof tagsToCompile === 'string')
        tagsToCompile = tagsToCompile.split(spaceRe, 2);
      // tagsToCompile 不是一个数组,或者 是数组,但是不是两位直接报错
      if (!isArray(tagsToCompile) || tagsToCompile.length !== 2)
        throw new Error('Invalid tags: ' + tagsToCompile);
      // 下面以解析符号获取正则表达式
      //  /\{\{\s*/   两个左大括号 后可能有空格 
      openingTagRe = new RegExp(escapeRegExp(tagsToCompile[0]) + '\\s*');
      //  /\s*\}\}/   两个右大括号 前可能有空格
      closingTagRe = new RegExp('\\s*' + escapeRegExp(tagsToCompile[1]));
      //  /\s*\}\}\}/ 三个右大括号 前可能有空格
      closingCurlyRe = new RegExp('\\s*' + escapeRegExp('}' + tagsToCompile[1]));
    }

    // 处理tags, 如果没有,那就用默认 tags ["{{", "}}"]
    compileTags(tags || mustache.tags);

    // 得到扫描器 Scanner类的示例
    var scanner = new Scanner(template);

    var start, type, value, chr, token, openSection;
    // 循环遍历 scanner
    while (!scanner.eos()) {
      // 获取 位置 pos 会根据scan 和scanUtil 变化
      start = scanner.pos;

      // 找到 {{ 开始符号前的文本
      value = scanner.scanUntil(openingTagRe);

      if (value) {
        // 把文本字符串 从第一个字符开始遍历 比如 "我是中国人   我爱中国"
        for (var i = 0, valueLength = value.length; i < valueLength; ++i) {
          // 获取当前字符  比如 第一个 我
          chr = value.charAt(i);
          // 如果当前字符为空字符 当前tokens的长度 放到spaces数组中
          // tokens每一项放的都是一个 字符,sapces会记录哪几项是空格
          //  把缩进长度增加当前空白字符
          if (isWhitespace(chr)) {
            spaces.push(tokens.length);
            indentation += chr;
          } 
            // 不是空白字符   
            else {
            nonSpace = true; // 设置 非空标识 为 ture
            lineHasNonSpace = true; // 设置当前行 标识为 true(有内容)
            indentation += ' '; // 缩进长度加上 一个空格 
          }

          // 把当前字符 以 数组放入tokens中 并记录当前位置和结束位置(指针)
          tokens.push([ 'text', chr, start, start + 1 ]);
          start += 1;

          // 遇到回车
          if (chr === '\n') {
            // 调用方法 去重空白
            stripSpace();
            indentation = ''; // 由于调用了去空白方法 indentation 重置为空
            tagIndex = 0; // tagIndex也重置为0
            lineHasNonSpace = false; // 当前行标志也重置为 false
          }
        }
      }

      // 如果没有 {{ , 说明已经找完了 结束循环
      if (!scanner.scan(openingTagRe))
        break;

      // 如果在 {{ 里面的内容 设置  hasTag 为true
      hasTag = true;

      // 处理 循环 注释等其他 类型
      // 根据 /#|\^|\/|>|\{|&|=|!/ 获取 如果没有匹配到,那就是默认的数据类型 name
      // 可能是 循环开始#  循环结束\ 注释!不用转义{ 等
      type = scanner.scan(tagRe) || 'name';
      // 跳过空字符串
      scanner.scan(whiteRe);

      // 如果type 为 = 符号 用来更改后面的解析符号用的tags
      if (type === '=') {
        // value 保存 新的 匹配符号  比如 {{=<% %>=}} 获取到 value = "<% %>"
        value = scanner.scanUntil(equalsRe);
        scanner.scan(equalsRe);
        scanner.scanUntil(closingTagRe);
      } else if (type === '{') {
        // value 或 }}} 前的内容  里面的内容不要转义
        value = scanner.scanUntil(closingCurlyRe);
        scanner.scan(curlyRe);
        scanner.scanUntil(closingTagRe);
        // 并且把类型重置为 & 符号
        type = '&';
      } else {
        // 其他情况 获取结束符号}} 前的文本
        value = scanner.scanUntil(closingTagRe);
      }

      // 匹配结束符号
      if (!scanner.scan(closingTagRe))
        throw new Error('Unclosed tag at ' + scanner.pos);

      // 如果 type 为 > 需要多记录 三个 参数
      // token [类型, 值, 开始位置, scanner的位置, 空白字符, 一行中遇到的标记数, 当前token有内容]
      if (type == '>') {
        token = [ type, value, start, scanner.pos, indentation, tagIndex, lineHasNonSpace ];
      } else {
        // 设置每一项token [类型, 值, 开始位置, scanner的位置]
        token = [ type, value, start, scanner.pos ];
      }
      
      // 标记数+1
      tagIndex++;
      // tokens 存入token
      tokens.push(token);

      // 如果遇到循环 或者判断 if else  需要把当前token 放入栈中(便于后面判断是否循环或判断结束)
      if (type === '#' || type === '^') {
        sections.push(token);
      } else if (type === '/') {
        // 遇到循环或判断  结束符号 取出 最后进的token
        openSection = sections.pop();

        // 没有内容报错
        if (!openSection)
          throw new Error('Unopened section "' + value + '" at ' + start);
        // 如果内容 [type, value, start, scanner.pos] 如果值不相等报错
        if (openSection[1] !== value)
          throw new Error('Unclosed section "' + openSection[1] + '" at ' + start);
      } else if (type === 'name' || type === '{' || type === '&') {
        // 如果type 为上面 的  那设置标志 为有内容
        nonSpace = true;
      } else if (type === '=') {
        // 为下一次循环 设置 新标签 可能改为了 <% %> 
        compileTags(value);
      }
    }

    // 循环结束 调用方法去除空白
    stripSpace();

    // 循环结束  判断栈中是否还有  内容
    openSection = sections.pop();

    if (openSection)
      throw new Error('Unclosed section "' + openSection[1] + '" at ' + scanner.pos);

    return nestTokens(squashTokens(tokens));
  }


  /**
   * 将给定“tokens”数组中连续文本标记的值合并为单个标记。
   * 把相邻的token 切 type都是text类型的文本合并为一个token
   */
  function squashTokens (tokens) {
    var squashedTokens = []; // 压缩后的数组

    var token, lastToken; // 当前token  和上一次 token
    for (var i = 0, numTokens = tokens.length; i < numTokens; ++i) {
      token = tokens[i];
      // [type, value, start, scanner.pos]

      if (token) {
        // 如果当前token类型为 text 。 并且上一个token(现在的lastToken) 类型也是text
        // 那么就把两个token合并
        // 这里的处理 把value 现在, scanner.pos 位置为现在token的位置
        if (token[0] === 'text' && lastToken && lastToken[0] === 'text') {
          lastToken[1] += token[1];
          lastToken[3] = token[3];
        } else {
          // 第一项 token 存起来, 并且赋值给lastToken
          // 或者现在的token和上一个token type不一样
          squashedTokens.push(token);
          lastToken = token;
        }
      }
    }

    return squashedTokens;
  }


  /**
   * 处理 # 开始 的循环接口。处理为嵌套的token
   * 把 type 为 # 的开始token ,结束为 '/'结束的token。 这一堆token作为数组,作为#token的第五项
   * 
   * 或者 处理  ^ /  类似于 if else 的判断
   * 
   * 这里要注意  收集器 的作用主要是 改变 一个指向。如要 # 就要把收集器指向 当前token的 第五项
   * 当遇到 / 说明循环结束了。需要把收集器指向上一级的token的第五项
   * 循环一直,只到没有循环为止,指向最初的token;
   */
  function nestTokens (tokens) {
    var nestedTokens = []; // 嵌套tokens
    var collector = nestedTokens; // 收集器 指向每一次需要收集的数组
    var sections = [];// 栈数组 用于 保存有多少次循环, 并存在循环的主token

    var token, section;
    for (var i = 0, numTokens = tokens.length; i < numTokens; ++i) {
      token = tokens[i];

      switch (token[0]) {
        // 如果type 为 # ^ 那就是嵌套开始项 或者 判断
        // [type, value, start, scanner.pos, [嵌套token]]
        case '#':
        case '^':
          // 收集器push 进token
          collector.push(token);
          // 栈中 推入当前token 这个是类似一个父token
          sections.push(token);
          // 收集器 现在指向 当前token 第五项并且为空数组
          // 只有遇到的token都要放入这个空数组中
          collector = token[4] = [];
          break;
        case '/':
          // 如果遇到/ 那么就是 循环或判断结束了。 
          // 结束了的话,把最后的一次栈中的token 取出来
          section = sections.pop();
          // 把 当前token 的scanner.pos 给 父token的第六项 
          section[5] = token[2];
          // 判断栈中还有没有数据。没有的话,就是初始nestedTokens;否则,
          // 收集器指针 需要指向上一级父token的第五位
          collector = sections.length > 0 ? sections[sections.length - 1][4] : nestedTokens;
          break;
        default:
          // 循环中 把每一个token 放入收集器中 (除开 循环或判断的情况)
          collector.push(token);
      }
    }

    return nestedTokens;
  }

  /**
   * 扫描器
   * 模板解析器用于在模板字符串中查找令牌的简单字符串扫描仪
   * 获取字符串,并设置尾部,尾巴会向后减少。pos是当前位置
   */
  function Scanner (string) {
    this.string = string;
    this.tail = string;
    this.pos = 0;
  }

  // 判断是否到尾部
  Scanner.prototype.eos = function eos () {
    return this.tail === '';
  };

  /**
   * 为了跳过匹配符号
   * 返回匹配的匹配的符号,如果没有返回空字符串
   * 比如 re为 {{   找到的话就返回 {{  并且尾巴向后移动两位,pos位置也加2
   */
  Scanner.prototype.scan = function scan (re) {
    var match = this.tail.match(re);
    // 没有匹配直接返回空字符串

    if (!match || match.index !== 0)
      return '';
    

    var string = match[0];
    
    // 尾巴 向后移动 匹配符号的长度

    this.tail = this.tail.substring(string.length);
    // 位置加 上匹配符号的长度
    this.pos += string.length;

    return string;
  };

  /**
   * 为了找到 匹配符号前的文本
   * 如果有匹配的符号,返回匹配符号之前的文本,否则就是没有匹配到,返回剩下的尾巴
   * 比如 这里是{{name}}的家  re为 {{  会返回 这里是
   */
  Scanner.prototype.scanUntil = function scanUntil (re) {
    // 得到匹配符号在尾巴中的位置(尾巴会变化,所有位置也会在变化)
    var index = this.tail.search(re), match;

    switch (index) {
      // -1说明没有匹配到特殊符号,说明已经扫描完了。直接返回剩下的文本。比把尾巴置空
      case -1:
        match = this.tail;
        this.tail = '';
        break;
        // 特殊符号刚好在开始地方,那匹配符号前 就没有文本。匹配到的就是空字符串
      case 0:
        match = '';
        break;
        // 其他情况 直接获取匹配符号前的文本,并把尾巴向后移动到匹配的地方(这个时候后面会用scan函数跳过匹配符号)
      default:
        match = this.tail.substring(0, index);
        this.tail = this.tail.substring(index);
    }

    // 位置 需要加上匹配的长度 ,并返回匹配的文本 
    this.pos += match.length;

    return match;
  };

  /**
   * 
   * view 类 数据 比如 {{name: 'zs', age: 14}}
   * 把数据作为参数 实例化Context 。便于 创建新的数据 并且根据树结构查找数据
   */
  function Context (view, parentContext) {
    this.view = view;
    // 缓存  对象  .  便于 循环  通过 {{.}} 找到值
    this.cache = { '.': this.view };
    // 设置父元素为 传入的第二个参数 (以便于 循环或者判断用,避免循环里面有变量名和根变量名重复; 比如 {name: 'zs', list:[{name: 'ls'}, {name: 'ww'}]})
    this.parent = parentContext;
  }

  // 使用给定视图创建一个新上下文,并将此上下文作为父级
  // push 方法 处理循环或判断用 ,创建一个新的contenxt实例,得到新的数据与模板
  Context.prototype.push = function push (view) {
    return new Context(view, this);
  };

  // 返回此上下文中给定名称的值,如果此上下文视图中缺少该值,则遍历上下文层次结构。(用来查找数据中的值)
  Context.prototype.lookup = function lookup (name) {
    // 首先拿到缓存
    var cache = this.cache;
    // 定义查找的内容

    var value;
    // 如果要查找的 内容在缓存中  赋值为value
    if (cache.hasOwnProperty(name)) {
      value = cache[name];
    } else {
      // context 初始定义为 当前对象,下面可能会改变
      // 增加一个标识 lookupHit 是否查找命中 
      // intermediateValue 中间值
      var context = this, intermediateValue, names, index, lookupHit = false;

      while (context) {
        // 如果要查找的是 a.b.c 类似 的数据
        if (name.indexOf('.') > 0) {
          // 首先把 数据赋值为 intermediateValue 比如 { a: { b: c: 'haha' } }
          intermediateValue = context.view;
          // names 把 a.b.c 拆分为 [a.b.c]
          names = name.split('.');
          index = 0;
          //  循环 names 数组 ,并分别从intermediateValue中查找到值,并重新赋值为intermediateValue
          //  比如上面最后会得到 intermediateValue 为 haha
          while (intermediateValue != null && index < names.length) {
            if (index === names.length - 1)
              lookupHit = (
                hasProperty(intermediateValue, names[index])
                || primitiveHasOwnProperty(intermediateValue, names[index])
              );

            intermediateValue = intermediateValue[names[index++]];
          }
        } else {
          // 直接返回匹配的值
          intermediateValue = context.view[name];
          // 对象自身属性中是否具有指定的属性(也就是,是否有指定的键)
          lookupHit = hasProperty(context.view, name);
        }

        // 如果有命中  把中间值赋值为value
        if (lookupHit) {
          value = intermediateValue;
          break;
        }

        context = context.parent;
      }

      // 缓存 name-value 的值
      cache[name] = value;
    }
    // 如果查找到的内容是 函数,然后执行到,赋值给value
    if (isFunction(value))
      value = value.call(this.view);

    return value;
  };

   /**
   * 提供解析模板为tokens 然后把tokens 转为 dom字符串
   * 根据tokens 转换为 字符串,并且缓存它
   */
  function Writer () {
    this.templateCache = {
      _cache: {},
      set: function set (key, value) {
        this._cache[key] = value;
      },
      get: function get (key) {
        return this._cache[key];
      },
      clear: function clear () {
        this._cache = {};
      }
    };
  }

  // 清空缓存
  Writer.prototype.clearCache = function clearCache () {
    if (typeof this.templateCache !== 'undefined') {
      this.templateCache.clear();
    }
  };

  // 解析和缓存给定的“模板”,并返回从解析生成的令牌数组。
  Writer.prototype.parse = function parse (template, tags) {
    // 拿到缓存
    var cache = this.templateCache;
    // 获取 改变后的 缓存key
    var cacheKey = template + ':' + (tags || mustache.tags).join(':');

    var isCacheEnabled = typeof cache !== 'undefined';
    // 获取从缓存中获取tokens 或在没有
    var tokens = isCacheEnabled ? cache.get(cacheKey) : undefined;

    if (tokens == undefined) {
      // 如果缓存中没有  需要重新 获取并缓存起来
      tokens = parseTemplate(template, tags);
      isCacheEnabled && cache.set(cacheKey, tokens);
    }
    return tokens;
  };

  /**
   * 渲染函数 
   * template 为模板
   * view 为数据
   * partials 为补充模板 可以为对象也可以为函数
   * config 为补充参数  里面可能有 自定义的解析符号tags  转义规则escape
   */
  Writer.prototype.render = function render (template, view, partials, config) {
    // 获取自定义转义符号 tags
    var tags = this.getConfigTags(config);
    // 获取 处理后的 tokens
    var tokens = this.parse(template, tags);
    // 把数据 view 用Context实例化, 便于储存数据和查找
    var context = (view instanceof Context) ? view : new Context(view, undefined);
    return this.renderTokens(tokens, context, partials, template, config);
  };

  // 递归函数  用于处理tokens  处理为dom字符串
  Writer.prototype.renderTokens = function renderTokens (tokens, context, partials, originalTemplate, config) {
    var buffer = '';

    var token, symbol, value;
    for (var i = 0, numTokens = tokens.length; i < numTokens; ++i) {
      value = undefined;
      // 获取每一个token
      token = tokens[i];
      // 获取token 的type 类型 
      symbol = token[0];

      // 不同的类型 用不同的方法处理
      // 循环
      if (symbol === '#') value = this.renderSection(token, context, partials, originalTemplate, config);
      // 判断
      else if (symbol === '^') value = this.renderInverted(token, context, partials, originalTemplate, config);
      // partials 提供 新的数据(接口返回的数据不够,需要自定义某些数据)
      else if (symbol === '>') value = this.renderPartial(token, context, partials, config);
      // 不用转义的
      else if (symbol === '&') value = this.unescapedValue(token, context);
      // 需要转义
      else if (symbol === 'name') value = this.escapedValue(token, context, config);
      // 普通文本
      else if (symbol === 'text') value = this.rawValue(token);

      if (value !== undefined)
        buffer += value;
    }

    return buffer;
  };

  // 如果是# 那这个token 就是循环的token  或者判断中的 if 语句
  // 比如 ["#", "list", 194, 203, Array(5), 260]
  Writer.prototype.renderSection = function renderSection (token, context, partials, originalTemplate, config) {
    var self = this;
    var buffer = '';
    // 数据中查找 list 对应的 数据
    var value = context.lookup(token[1]);

    // 附属渲染  获取  顶层渲染的数据  self 当前 最顶层
    function subRender (template) {
      return self.render(template, context, partials, config);
    }
    // 如果数据没有则跳过

    if (!value) return;

    // 如果是 数组
    // 比如  {list: [{name:'zs'},{name: 'ls'}]}
    // value 为[{name:'zs'}, {name: 'ls'}, {name: 'zl'}]
    if (isArray(value)) {
      for (var j = 0, valueLength = value.length; j < valueLength; ++j) {
        // 递归 把 Array(5) 作为token, 每一项 {name: 'xx'} 作为数据
        // 调用 context.push 方法传入 对应的数据 value[j]
        buffer += this.renderTokens(token[4], context.push(value[j]), partials, originalTemplate, config);
      }
      //如果 找到的 value 为对象或字符串
      // {list: {name:'zs', age:14}} 或者 {list: 'abcd'} 或者 {list: 1234}
      // value 为 {name:'zs', age:14} 或 'abcd' 或1234
    } else if (typeof value === 'object' || typeof value === 'string' || typeof value === 'number') {
      // 也是递归 把 Array(5)作为token, 直接把当前value作为数据
      // 调用 context.push 方法传入 对应的数据 value
      buffer += this.renderTokens(token[4], context.push(value), partials, originalTemplate, config);
    // 如果找到的函数
    /**
     *  arr: ()=> (template, fn)=>{
     *     return fn(template)
     *   }
     * 
     *  template  == originalTemplate.slice(token[3], token[5]) 为循环内的位解析字符串
     * fn == subRender 用根 数据渲染当前模板
     */
    } else if (isFunction(value)) {
      // 如果 原始模板不是字符串 则报错
      if (typeof originalTemplate !== 'string')
        throw new Error('Cannot use higher-order sections without the original template');

      // 返回一个自定义函数, 原函数匹配的字符串为第一个参数 ,subRender为第二个参数(以根数据来渲染当前模板)
      value = value.call(context.view, originalTemplate.slice(token[3], token[5]), subRender);

      if (value != null)
        buffer += value;
    } else {
      // 其他情况 直接调用 递归处理 数据 就是当前 context
      // 是 判断语句,如果找到就直接渲染拼接 ,没有会返回空
      buffer += this.renderTokens(token[4], context, partials, originalTemplate, config);
    }
    return buffer;
  };

  // 判断 类似于 if else  中的 else 
  // {{# flag}}flag为ture显示{{/flag}}{{^flag}}flag为false显示{{/flag}}
  Writer.prototype.renderInverted = function renderInverted (token, context, partials, originalTemplate, config) {
    // 得到 value 需要判断的内容
    var value = context.lookup(token[1]);

    // 如果没有 或者是个空数组  
    if (!value || (isArray(value) && value.length === 0))
      return this.renderTokens(token[4], context, partials, originalTemplate, config);
  };

  // 这个函数为啥要这么处理,,,我没有尝试出来。。。。
  Writer.prototype.indentPartial = function indentPartial (partial, indentation, lineHasNonSpace) {

    var filteredIndentation = indentation.replace(/[^ \t]/g, ''); // 除了空格 和 制表符TAB 的其他空白 都去掉
    
    var partialByNl = partial.split('\n'); // 把模板 按 换行符 切割为数组
    
    for (var i = 0; i < partialByNl.length; i++) {
      // 模板有内容 并且  (没有内容 或者  不是第一项)
      if (partialByNl[i].length && (i > 0 || !lineHasNonSpace)) {
        // 模板 替换为  处理后的filteredIndentation + 当前模板
        partialByNl[i] = filteredIndentation + partialByNl[i];
      }
    }
    // 处理之后把 模板 按 换行符 拼接起来
    return partialByNl.join('\n');
  };

  // 根据 补充数据 partials 获取数据 比如  Mustache.render("<div>{{>abc}}</div>", data, { abc: '123', bbb: 'ttt' });
  // 栗子: partials = {abc: '123', bbb: 'ttt' }
  // 比如 token  [">", "abc", 12, 42, "           ", 0, true]
  Writer.prototype.renderPartial = function renderPartial (token, context, partials, config) {
    
    // 没有数据源 就返回
    if (!partials) return;
    // 获取自动以的 标签tags
    var tags = this.getConfigTags(config);

    // 先判断 partials 是不是 函数 ,函数直接调用,否则直接返回 新模板
    var value = isFunction(partials) ? partials(token[1]) : partials[token[1]];
    if (value != null) {
      // token token [类型, 值, 开始位置, scanner的位置, 空白字符, 一行中遇到的标记数, 当前token有内容]
      var lineHasNonSpace = token[6]; // 当前token 有内容 标志
      var tagIndex = token[5]; // 一行中遇到的标记数 (碰到换行符 会重置为0)
      var indentation = token[4]; // 空白字符
      var indentedValue = value; // 比如: 123 
      
      // 标记数为0 (新起的一行 )并且有内容
      if (tagIndex == 0 && indentation) {
        indentedValue = this.indentPartial(value, indentation, lineHasNonSpace);
      }
      
      // 获取处理后的tokens  比如  ['text', 123, 0, 3]
      var tokens = this.parse(indentedValue, tags);
     
      return this.renderTokens(tokens, context, partials, indentedValue, config);
    }
  };

  // 获取 token里面的值 不用转义
  Writer.prototype.unescapedValue = function unescapedValue (token, context) {
    var value = context.lookup(token[1]);
    if (value != null)
      return value;
  };

  // 获取 token里面的值 并 转义 (转义规则可能由用户传入)
  Writer.prototype.escapedValue = function escapedValue (token, context, config) {
    // 获取 用户传入的转义规则 没有 就用默认的转义规则 
    var escape = this.getConfigEscape(config) || mustache.escape;

    var value = context.lookup(token[1]);
    if (value != null)
      // 如果得到的内容 是数字 或者就 等于转义后的字符串  就直接返回当前内容, 否则需要调用 escape方法转义后返回
      return (typeof value === 'number' && escape === mustache.escape) ? String(value) : escape(value);
  };

  // 获取 未加工的值 (保存的 文本)
  Writer.prototype.rawValue = function rawValue (token) {
    return token[1];
  };

  // 获取config 的tags (自定义标签 tags)
  Writer.prototype.getConfigTags = function getConfigTags (config) {
    // 是数组返回 本身
    if (isArray(config)) {
      return config;
    }
    // 如果是对象,返回 对象的tags属性
    else if (config && typeof config === 'object') {
      return config.tags;
    }
    // 否则 返回undefined
    else {
      return undefined;
    }
  };

  // 获取 config 的转义符号对象 (自定义转义规则)
  Writer.prototype.getConfigEscape = function getConfigEscape (config) {
    if (config && typeof config === 'object' && !isArray(config)) {
      return config.escape;
    }
    else {
      return undefined;
    }
  };

  var mustache = {
    name: 'mustache.js',
    version: '4.1.0',
    tags: [ '{{', '}}' ],
    clearCache: undefined,
    escape: undefined,
    parse: undefined,
    render: undefined,
    Scanner: undefined,
    Context: undefined,
    Writer: undefined,

    /**
     * 允许用户通过为对象提供set、get和clear方法来覆盖默认的缓存策略。也可以通过将缓存设置为文字“undefined”来禁用缓存。
     */
    set templateCache (cache) {
      defaultWriter.templateCache = cache;
    },
    /**
     * 获取缓存
     */
    get templateCache () {
      return defaultWriter.templateCache;
    }
  };

  // 实例化 Writer
  var defaultWriter = new Writer();

  /**
   * 清除缓存
   */
  mustache.clearCache = function clearCache () {
    return defaultWriter.clearCache();
  };

  /**
   * 把模板处理为tokens
   */
  mustache.parse = function parse (template, tags) {
    return defaultWriter.parse(template, tags);
  };

  /**
   * 暴露的render 方法为实例化Writer之后的render方法
   */
  mustache.render = function render (template, view, partials, config) {
    // template 模板必须为字符串
    if (typeof template !== 'string') {
      throw new TypeError('Invalid template! Template should be a "string" ' +
                          'but "' + typeStr(template) + '" was given as the first ' +
                          'argument for mustache#render(template, view, partials)');
    }

    return defaultWriter.render(template, view, partials, config);
  };

  // escape 作用是把特殊符号转移
  mustache.escape = escapeHtml;

  // 导出三个类主要用于测试,但也用于高级用途
  mustache.Scanner = Scanner;
  mustache.Context = Context;
  mustache.Writer = Writer;

  return mustache;

})));

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,776评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,527评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,361评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,430评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,511评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,544评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,561评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,315评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,763评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,070评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,235评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,911评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,554评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,173评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,424评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,106评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,103评论 2 352

推荐阅读更多精彩内容