包装一系列闭包

Wrapping a series of closures

我继承了一大段代码,其中有一大组 DataTable 渲染器。我想在前面的数据对象中插入一些额外的数据而不重写所有渲染器,只重写那些实际受影响的渲染器。

columnDefs 字段生成为具有 render 函数的对象数组,通常还有其他字段(targetstype 等) .我想即时重写它,使 render 函数成为调用原始函数的包装器。我找不到正确的代码来关闭原始函数,因此我可以在处理过程中修改数据并仍然调用该原始函数。

例如,给定一个 columnDefs 为:

var columnDefs = [
{
  "render": function (data, type, row) {
    return stuff(row[3]);
  },
  "targets": 2
},
{
  "render": function (data, type, row) {
    return stuff(row[0]);
  },
  "targets": 1
},
// etc.
]

我正在尝试做这样的事情:

var mapper = function (def) {
    (function (d, old_render) {
        d.render = (data, type, row) => {
            let extra_data = undefined;
            if (row.length > 1 && typeof (row[0]) == 'object') {
                extra_data = row[0];
                row = row.slice(1);
            }
            return old_render(data, type, row, extra_data);
        }
    })(def, def.render);
};
columnDefs.forEach(mapper);

在调用 $("#my-datatable").DataTable({ ..., columnDefs: columnDefs, ... }).

之前

目前有效的是正在调用新的渲染函数。但是,当调用 render 时,old_render 始终未定义,这当然会死亡。

完整的代码示例 on JSFiddle 为了方便起见,也在这里:

function stuff(x) {
  return "<b>" + x + "</b>";
}

var columnDefs = [
{
  "render": function (data, type, row) {
    return stuff(row[3]);
  },
  "targets": 2
},
{
  "render": function (data, type, row) {
    return stuff(row[0]);
  },
  "targets": 1
},
]

var dataSource = [{"a", "b", "c", "d", "e"}, {"f","g","h","i","j"}];
columnDefs.forEach(d => {
  d.render = function(old_render) {
    return (data, type, row) => {
      let extra_data = undefined;
      return old_render(data, type, row.map(, extra_data);
    };
  }(d.render);
});
var table = $("#my-table").DataTable({
processing: true,
data: dataSource,
columnDefs: columnDefs
});

而关联的 HTML 只是:

<link href="//cdn.datatables.net/1.10.19/css/jquery.dataTables.min.css" rel="stylesheet" type="text/css" />
<script src="https://cdn.datatables.net/1.10.19/js/jquery.dataTables.min.js"></script>

<p>
hello
</p>
<table id="my-table">
<thead>
  <tr>
    <th>1</th><th>2</th><th>3</th><th>4</th>
  </tr>
</thead>
</table>

更新...JSLint 正在运行,我只是不确定为什么它不在我较大的代码库中。我想回到绘图板,找出原因。

像这样的东西应该有用...

function newRender(oldRender, data, type, row) {
    let extra_data = undefined;
    if (row.length > 1 && typeof (row[0]) == 'object') {
        extra_data = row[0];
        row = row.slice(1);
    }
    return oldRender(data, type, row, extra_data);
}

var mapper = function (def) {
    def.render = newRender.bind(null, def.render);
};

columnDefs.forEach(mapper);

事实证明,如果某些记录没有输入渲染器,那么当然无法调用旧渲染器。这里处理的列数足够多,我没有注意到其中一些没有渲染器。

var columnDefs = [
{
  "render": function (data, type, row) {
    return stuff(row[3]);
  },
  "targets": 2
},
{
  //"render": function (data, type, row) {
  //  return stuff(row[0]);
  //},
  "targets": 1
},
]

现在当代码到达此处的第二列时,没有渲染函数,因此当然是未定义的。通过用 if (d.render) {...} 包装映射器,问题就这样避免了。