【问题标题】:jQuery: force display of modified domjQuery:强制显示修改后的dom
【发布时间】:2013-01-09 11:03:47
【问题描述】:

我在尝试在我的页面上设置“加载微调器”时遇到了问题,该页面在对表格进行排序时运行,尤其是对于速度较慢的客户端,因为可能需要 10 秒才能对页面进行排序。我可以看到 DOM 被微调器代码修改,但它不显示。我希望在排序发生之前我可以做些什么来强制显示微调器,当然在排序完成时停止它。

我的排序基于'sorttable.js',我已对其进行了修改以处理表第一列(其中包含名称)的二级排序。

我的微调器使用'spin.js'

我还是这个 jQuery 的新手,而且这个可排序的代码相当复杂。我在下面突出显示部分,但我的完整修改后的可排序代码(现在)可以在'sorttable-TESTING-ONLY.js' 找到,测试 html 页面在'TESTING-ONLY-sort_and_spin.htm'

因此,脚本会在页面加载时设置一些功能(“.....”表示跳过的行):

makeSortable: function(table) {                          //line 75
  ......
  headrow = table.tHead.rows[0].cells;

  for (var i=0; i<headrow.length; i++) {                 //line 114
    .....
    headrow[i].sorttable_sortfunction = sorttable.guessType(table,i);   //line 126

    // code to start/stop spinner
    headrow[i].sorttable_spinner = (function(val) {      //line 136
      return function(val) {
        var $this = $(this),
        data = $this.data();

        if (data.spinner) {
          data.spinner.stop();
          delete data.spinner;
        }
        if (val > 0) {
          var opts = {
            .......
           };
           data.spinner = new Spinner($.extend({color: $this.css('color')}, opts)).spin(this);
           // can see spinner added to DOM but does not display...
         }
    }})(i);

对它们进行编码会为列标题创建一个可点击的事件处理程序:

headrow[i].sorttable_columnindex = i;                            //line 171
headrow[i].sorttable_tbody = table.tBodies[0];

dean_addEvent(headrow[i],"click", function(e) {                  //line 176
   .... does some stuff with class names, etc

   this.sorttable_spinner(1);    // set spinner on             //line 224

   .... builds array to do sort then calls sort functions
   if (sort_direction == 'forward') {                            //line 247
     row_array.sort(this.sorttable_sortfunction); 
   } else {
     row_array.sort(this.sorttable_reversesortfunction);
   }

   tb = this.sorttable_tbody;
   for (var j=0; j<row_array.length; j++) {
     tb.appendChild(row_array[j][2]);
   }
   delete row_array;

   this.sorttable_spinner();  // now stop the spinner            //line 261

这一切看起来应该可以正常工作。 在带有 firebug 的 Firefox 中,我看到 DOM 加载了微调器代码,我看到它在浏览器中旋转,并且关闭微调器的行将其从 DOM 中删除。 但是,如果我没有处于调试模式,只运行它,那么不会显示微调器? (用 IE10 调试甚至不显示微调器)。

我在不同的地方尝试过 .show() 但总是被告知没有可用的功能。 看到对 window.setTimeout( function () {... 的引用以提供单独的排序过程,但不明白在这种情况下如何实现。

如果有人能给我一些建议,将不胜感激。

问候,布莱斯 S.

【问题讨论】:

  • 顺便说一句 - 由于假期超出互联网范围,因此无法在 5 天内查看答案 :-)
  • 我想我会在这里留言以查看我的fork of tablesorter。我试图在 jsFiddle 上为您制作一个演示,但它无法处理 3800 多个表格行,但在我的桌面上我发现排序大约需要 3 秒。如果您不使用 zebra 小部件并使用 css (tbody tr:nth-child(odd)) 应用行着色,它会更快。
  • 嗨,Mottie,感谢您提供的链接 - 我希望我在几周前就找到了。我的 googleFoo 让我失望了。我会继续我现在所拥有的。干杯,布莱斯。

标签: javascript jquery tablesorter spin.js


【解决方案1】:

基本上,JavaScript 只有一个线程用于您的代码和 UI。这意味着,如果您的代码不放弃控制权,那么在您放弃控制权之前不会显示 UI。所以,在这种代码中:

spinner_on();
dostuff();
spinner_off();

发生的情况是,您将微调器插入 DOM,执行操作,移除微调器,放弃控制 - 最后更新 UI,没有微调器(因为此时,它不再存在)。

基本模式应该是这样的:

spinner_on();
setTimeout(function() {
  dostuff();
  spinner_off();
}, 0);

它的工作原理如下:将微调器插入 DOM,安排一些事情,放弃控制。 UI 得到更新(使用微调器)。计划函数运行:事情完成,微调器从 DOM 中移除,控制权再次被放弃。 UI 已更新(没有微调器)。

当您在调试器中时,调试器会从您的代码手中夺走控制权,因此您可以在代码中断时在 UI 中看到您的微调器。

【讨论】:

  • 嗨,Amadan,太好了,我现在可以看到微调器了。在我启动微调器(第 224 行)之后,在上面的 dean_addEvent 函数中添加了 setTimeout(),使其余代码成为 setTimeout 调用的一部分。但是,我看到了我的微调器,但它没有旋转 - 它在第一次显示时是静态的。有什么办法可以解决吗?
  • 我可以设置一些东西在代码和 UI 之间交换,以便屏幕刷新。或者让两个线程以某种方式运行......
  • 正如我所说,JavaScript 只在一个线程中运行。 (实际上,您可以使用 WebWorkers 获得更多线程,但它们无法触摸 UI,因此您无法使用它们。)使您的代码允许屏幕刷新的唯一方法是如果您更频繁地插入 setTimeout - 例如,摆脱你的循环,并使用setTimeout 让代码自己调用。
  • 谢谢。我将不得不查看只是弹出一条消息说“排序...”,而不是作为我主要耗时的“循环”,我认为我无法访问,因为它是作为内置的一部分完成的排序”功能。干杯,布莱斯。
  • 你有多少行?您是否对sortappendRow 循环进行了计时?但是,是的,静态消息可能是个好主意。
猜你喜欢
  • 2019-08-25
  • 1970-01-01
  • 2012-10-15
  • 1970-01-01
  • 1970-01-01
  • 2012-03-31
  • 1970-01-01
  • 2014-11-09
  • 2016-01-20
相关资源
最近更新 更多