扩展

𝑔𝑜𝑜𝑔𝑥ℎ2019年11月5日
大约 5 分钟

当我们使用 jQuery 对象的方法时,由于 jQuery 对象可以操作一组 DOM,而且支持链式操作,所以用起来非常方便。

但是 jQuery 内置的方法永远不可能满足所有的需求。比如,我们想要高亮显示某些 DOM 元素,用 jQuery 可以这么实现:

$("span.hl").css("backgroundColor", "#fffceb").css("color", "#d85030");

$("p a.hl").css("backgroundColor", "#fffceb").css("color", "#d85030");

总是写重复代码可不好,万一以后还要修改字体就更麻烦了,能不能统一起来,写个 highlight() 方法?

$("span.hl").highlight();

$("p a.hl").highlight();

答案是肯定的。我们可以扩展 jQuery 来实现自定义方法。将来如果要修改高亮的逻辑,只需修改一处扩展代码。这种方式也称为编写 jQuery 插件。

编写 jQuery 插件

给 jQuery 对象绑定一个新方法是通过扩展 $.fn 对象实现的。让我们来编写第一个扩展—— highlight1():

$.fn.highlight1 = function () {
  // this已绑定为当前jQuery对象:
  this.css("backgroundColor", "#fffceb").css("color", "#d85030");
  return this;
};

注意到函数内部的 this 在调用时被绑定为 jQuery 对象,所以函数内部代码可以正常调用所有 jQuery 对象的方法。

对于如下的 HTML 结构:

<!-- HTML结构 -->
<div id="test-highlight1">
  <p>什么是<span>jQuery</span></p>
  <p><span>jQuery</span>是目前最流行的<span>JavaScript</span>库。</p>
</div>

来测试一下 highlight1() 的效果:

"use strict";

$("#test-highlight1 span").highlight1();

细心的您可能发现了,为什么最后要 return this ;? 因为 jQuery 对象支持链式操作,我们自己写的扩展方法也要能继续链式下去:

$("span.hl").highlight1().slideDown();

不然,用户调用的时候,就不得不把上面的代码拆成两行。

但是这个版本并不完美。有的用户希望高亮的颜色能自己来指定,怎么办?

我们可以给方法加个参数,让用户自己把参数用对象传进去。于是我们有了第二个版本的 highlight2():

$.fn.highlight2 = function (options) {
  // 要考虑到各种情况:
  // options为undefined
  // options只有部分key
  const bgcolor = (options && options.backgroundColor) || "#fffceb";
  const color = (options && options.color) || "#d85030";
  this.css("backgroundColor", bgcolor).css("color", color);
  return this;
};

对于如下 HTML 结构:

<!-- HTML结构 -->
<div id="test-highlight2">
  <p>什么是<span>jQuery</span> <span>Plugin</span></p>
  <p>
    编写<span>jQuery</span>
    <span>Plugin</span>可以用来扩展<span>jQuery</span>的功能。
  </p>
</div>

来实测一下带参数的 highlight2():

"use strict";

$("#test-highlight2 span").highlight2({
  backgroundColor: "#00a8e6",
  color: "#ffffff",
});

对于默认值的处理,我们用了一个简单的 &&|| 短路操作符,总能得到一个有效的值。

另一种方法是使用 jQuery 提供的辅助方法 $.extend(target, obj1, obj2, ...),它把多个 object 对象的属性合并到第一个 target 对象中,遇到同名属性,总是使用靠后的对象的值,也就是越往后优先级越高:

// 把默认值和用户传入的options合并到对象{}中并返回:
const opts = $.extend(
  {},
  {
    backgroundColor: "#00a8e6",
    color: "#ffffff",
  },
  options
);

紧接着用户对 highlight2() 提出了意见: 每次调用都需要传入自定义的设置,能不能让我自己设定一个缺省值,以后的调用统一使用无参数的 highlight2()?

也就是说,我们设定的默认值应该能允许用户修改。

那默认值放哪比较合适? 放全局变量肯定不合适,最佳地点是 $.fn.highlight2 这个函数对象本身。

于是最终版的 highlight() 终于诞生了:

$.fn.highlight = function (options) {
  // 合并默认值和用户设定值:
  const opts = $.extend({}, $.fn.highlight.defaults, options);
  this.css("backgroundColor", opts.backgroundColor).css("color", opts.color);
  return this;
};

// 设定默认值:
$.fn.highlight.defaults = {
  color: "#d85030",
  backgroundColor: "#fff8de",
};

这次用户终于满意了。用户使用时,只需一次性设定默认值:

$.fn.highlight.defaults.color = "#fff";
$.fn.highlight.defaults.backgroundColor = "#000";

然后就可以非常简单地调用 highlight() 了。

对如下的 HTML 结构:

<!-- HTML结构 -->
<div id="test-highlight">
  <p>如何编写<span>jQuery</span> <span>Plugin</span></p>
  <p>
    编写<span>jQuery</span>
    <span>Plugin</span
    >,要设置<span>默认值</span>,并允许用户修改<span>默认值</span>,或者运行时传入<span>其他值</span></p>
</div>

实测一下修改默认值的效果:

"use strict";

$.fn.highlight.defaults.color = "#659f13";
$.fn.highlight.defaults.backgroundColor = "#f2fae3";

$("#test-highlight p:first-child span").highlight();

$("#test-highlight p:last-child span").highlight({
  color: "#dd1144",
});

最终,我们得出编写一个 jQuery 插件的原则:

  • $.fn 绑定函数,实现插件的代码逻辑;
  • 插件函数最后要 return this ;以支持链式调用;
  • 插件函数要有默认值,绑定在 $.fn.<pluginName>.defaults 上;
  • 用户在调用时可传入设定值以便覆盖默认值。

针对特定元素的扩展

我们知道 jQuery 对象的有些方法只能作用在特定 DOM 元素上,比如 submit() 方法只能针对 form。如果我们编写的扩展只能针对某些类型的 DOM 元素,应该怎么写?

还记得 jQuery 的选择器支持 filter() 方法来过滤吗? 我们可以借助这个方法来实现针对特定元素的扩展。

举个例子,现在我们要给所有指向外链的超链接加上跳转提示,怎么做?

先写出用户调用的代码:

$("#main a").external();

然后按照上面的方法编写一个 external 扩展:

$.fn.external = function () {
  // return返回的each()返回结果,支持链式调用:
  return this.filter("a").each(function () {
    // 注意: each()内部的回调函数的this绑定为DOM本身!
    const a = $(this);
    const url = a.attr("href");
    if (
      url &&
      (url.indexOf("http://") === 0 || url.indexOf("https://") === 0)
    ) {
      a.attr("href", "#0")
        .removeAttr("target")
        .append(' <i class="uk-icon-external-link"></i>')
        .click(function () {
          if (confirm("您确定要前往" + url + "? ")) {
            window.open(url);
          }
        });
    }
  });
};

对如下的 HTML 结构:

<!-- HTML结构 -->
<div id="test-external">
  <p>如何学习<a href="https://jquery.com">jQuery</a>?</p>
  <p>
    首先,您要学习<a
      href="/wiki/001434446689867b27157e896e74d51a89c25cc8b43bdb3000"
      >JavaScript</a
    >,并了解基本的<a href="https://developer.mozilla.org/en-US/docs/Web/HTML"
      >HTML</a
    ></p>
</div>

实测外链效果:

"use strict";

$("#test-external a").external();

小结

扩展 jQuery 对象的功能十分简单,但是我们要遵循 jQuery 的原则,编写的扩展方法能支持链式调用、具备默认值和过滤特定元素,使得扩展方法看上去和 jQuery 本身的方法没有什么区别。