JQuery 可排序列表和固定/锁定项目

Posted

技术标签:

【中文标题】JQuery 可排序列表和固定/锁定项目【英文标题】:JQuery sortable lists and fixed/locked items 【发布时间】:2011-05-17 00:02:17 【问题描述】:

是否可以锁定 JQuery 可排序列表中的列表项,以使这些项目保留在列表中的特定位置。

例如,

考虑这个带有锁定项目的伪列表...

item A
item B(locked)
item C(locked)
item D
item E
item F
item G(locked)

所以,我希望将项目 B、C 和 G 固定为这样,如果用户将项目 D 拖放到列表的开头,则项目 A“跳过”固定/锁定的项目B 和 C 的结果如下...

item D
item B(locked)
item C(locked)
item A
item E
item F
item G(locked)

我一直在寻找这样的东西没有运气。有可能吗..?

【问题讨论】:

Perpaps 取消绑定列表项的所有事件,并仅重新绑定未锁定的事件? 您能否展示您用于处理可排序列表的 jQuery 代码,或者拼凑一个 JS Fiddle 或 JS Bin 演示? 【参考方案1】:

看看这个:Forcing an item to remain in place in a jQuery UI Sortable list

另外,我在这里用多个固定元素实现了上述解决方案:http://jsfiddle.net/enBnH/12/(已过时,见下文) 我认为这是不言自明的。

编辑:

我已经自动化了生成 lockto 值以及将 ID 添加到具有“固定”类的 lis 的过程(请注意,我必须添加一个 ID,以便我们可以引用它)

在此处查看完整的解决方案:http://jsfiddle.net/enBnH/44/

编辑

好的,在上面出现无数错误之后,我自己重写了该死的东西: http://jsfiddle.net/thomas4g/GPMZZ/15/

注意:以上方法确实有效,但@DarthJDG 的回答对我来说似乎好多了。我要让我的人可能更喜欢我的行为方式(我学会了不要删除东西,只是因为有更好的版本:P)

【讨论】:

感谢您的回答,将检查并报告 我认为这是一个不同的问题。 @omnosis 怎么会这样?在我看来是一样的。 @omnosis lockto 变量必须与固定元素的 id 匹配。请参阅此工作演示:jsfiddle.net/enBnH/8 这只适用于同时一个lockto#static-1 id 也在污染 id-s。顺便说一句很好的演示。【参考方案2】:

这是一个希望没有错误的版本,在您拖动时会更新。它会在排序开始时生成项目的当前所需位置,这意味着您应该能够在需要时更改类,刷新小部件的列表项,然后就可以开始了。

它还使用 sortable 的内置 items 属性来防止拖动固定项并解决列表顶部和底部的任何排序问题。

我试图移动固定项目,但这导致了可怕的错误行为,尤其是当组中有多个固定项目时。最终的解决方案是从列表中分离所有固定项,在前面添加一个辅助元素,然后将固定元素重新插入到所需位置,这似乎修复了所有错误。

在此处试用演示:http://jsfiddle.net/PQrqS/1/

html

<ul id="sortable">
    <li>oranges</li>
    <li class="static">apples</li>
    <li>bananas</li>
    <li>pineapples</li>
    <li>grapes</li>
    <li class="static">pears</li>
    <li>mango</li>
</ul>

CSS:

.static  color:red; 

li  background-color:whitesmoke; border:1px solid silver; width:100px; padding:2px; margin:2px; 

javascript

$('#sortable').sortable(
    items: ':not(.static)',
    start: function()
        $('.static', this).each(function()
            var $this = $(this);
            $this.data('pos', $this.index());
        );
    ,
    change: function()
        $sortable = $(this);
        $statics = $('.static', this).detach();
        $helper = $('<li></li>').prependTo(this);
        $statics.each(function()
            var $this = $(this);
            var target = $this.data('pos');

            $this.insertAfter($('li', $sortable).eq(target));
        );
        $helper.remove();
    
);

【讨论】:

@Thomas Shields:不高兴,这太可怕了,赏金仍在等待中。 :) 希望这次我能做到。 不错。 “手感”和操作也比其他方法好。 @Alex:感谢您的慷慨赏金。 辅助对象只是在位置 0 插入。事实上,这从来没有必要,因为如果位置 0 的项目不可排序,Sortable 将永远不会移动它。因此,您可以只检查位置 0(或强制检查所需位置是否与当前位置不同)并跳过它。然后你可以删除助手。【参考方案3】:

我扩展了jQuery.Ui.sortable

概述

jQuery.Ui.sortable 具有fixed 功能的小部件扩展。此功能允许用户修复列表中的元素。 使用.fixedsortable() 构造函数,您可以构造一个.sortable() 类,该类扩展了这些特性。您可以使用原始方法和扩展方法。

代码

https://gist.github.com/3758329#file_fixedsortable.js > fixedsortable.js

示例

http://jsfiddle.net/omnosis/jQkdb/

用法

一般:

要使用,请将fixed 属性添加到可排序列表选项中:

$("#list").fixedsortable(
   fixed: (value)
)

值可以是:

整数 示例:3 数组 整数示例:[1,2,5] 一个html元素或html元素列表 css 选择器 jquery 对象

HTML:

<script type="text/javascript" src="https://ajax.googleapis.com/ajax/libs/jquery/1.6.1/jquery.min.js"></script> //the jquery 
<script type="text/javascript" src="https://ajax.googleapis.com/ajax/libs/jqueryui/1.8.13/jquery-ui.min.js"></script> //the original jquery-ui   
<script type="text/javascript" src="https://raw.github.com/gist/3758329/91749ff63cbc5056264389588a8ab64238484d74/fixedsortable.js"></script> //the extended sortable
...
<ul id="sortable1">
    <li>oranges</li>
    <li class="static">apples</li>
    <li>bananas</li>
    <li>pineapples</li>
    <li>grapes</li>
    <li class="static">pears</li>
    <li>mango</li>
</ul>

<ul id="sortable2">
    <li>bananas</li>
    <li foo="asd">oranges</li>
    <li foo="dsa">apples</li>
    <li>pineapples</li>
    <li>grapes</li>
    <li>pears</li>
    <li>mango</li>
</ul>

<ul id="sortable3">
    <li>bananas</li>
    <li>oranges</li>
    <li>apples</li>
    <li>pineapples</li>
    <li>grapes</li>
    <li>pears</li>
    <li>mango</li>
</ul>

Javascript

$(function() 
    $("#sortable1").fixedsortable(
        fixed: "> .static"
    );

    $("#sortable2").fixedsortable(
        fixed: $("li[foo]").css("background","red")
    );

    $("#sortable3").fixedsortable(
        fixed: 2
    )
);

注意事项:

如果你坚持使用.sortable 代替.fixedsortable,你可以使用https://gist.github.com/3758329#file_sortable.js 代替jquery.ui 库。这是jQuery.ui 的完全替代品,但我不建议使用它,因为以后会更新。

我已经为此工作了超过 12 个小时 :( 我疯了..

【讨论】:

干得好!很好的答案,很好的布局,小提琴链接,解释!干得好。 干得好!我注意到,如果我正在排序的列表项包含链接(例如,fruit),则该行为与预期不符。移动非固定列表项可以找到,但“固定”项可以移动到可排序中的新位置。固定项目的原始位置保留了没有新项目排序到其中的行为。查看你的 jsfiddle 的这个分支:jsfiddle.net/jbru/wphpG/1 这个脚本还在某处可用吗?该链接不再指向任何地方。谢谢。【参考方案4】:

使用items 参数,您可以像这样实现您想要的:

$("#mytable tbody").sortable(items: 'tr.sortable');

现在只能对具有.sortable CSS 类的行进行排序。

如果您只想锁定第一行,您可以这样做:

$("#mytable tbody").sortable(items: 'tr:not(:first)');

可能性无穷无尽……

【讨论】:

这不起作用。 Here's why。抓住“一个”并将其向下拖动。请注意,“二”在“一”上方移动,此时它应保留为列表中的第二项。【参考方案5】:

这是基于@DarthJDG 代码。然而,它并没有检索到所有的 id,并且排序不适用于表格。所以我设法更新了他的解决方案,它适用于列表和表格,并将 id 保存在数组中。

Javascript:

var fixed = '.static'; //class which will be locked
var items = 'li'; //tags that will be sorted

$('ul').sortable(
  cancel: fixed,
  items: items,
  start: function () 
    $(fixed, this).each(function () 
      var $this = $(this);
      $this.data('pos', $this.index());
    );
  ,
  change: function () 
    var $sortable = $(this);
    var $statics = $(fixed, this).detach();
    var tagName = $statics.prop('tagName');
    var $helper = $('<'+tagName+'/>').prependTo(this);
    $statics.each(function () 
      var $this = $(this);
      var target = $this.data('pos');
      $this.insertAfter($(items, $sortable).eq(target));
    );
    $helper.remove();
  
);

演示:http://plnkr.co/edit/hMeIiRFT97e9FGk7hmbs

【讨论】:

【参考方案6】:

哦不!要点链接已损坏。这是来自https://gist.github.com/peterh-capella/4234752的代码转储

代码于 2016 年 1 月 6 日访问

//this code is created to fix this problem: http://***.com/questions/4299241/

(function( $, undefined ) 

$.widget("ui.fixedsortable", $.ui.sortable, 

    options: $.extend(,$.ui.sortable.prototype.options,fixed:[]),

    _create: function() 
      var o = this.options;
      this.containerCache = ;
      this.element.addClass("ui-sortable");

      //Get the items
      $.ui.sortable.prototype.refresh.apply(this,arguments);

      if( typeof this.options.fixed == "number") 
        var num = this.options.fixed
        this.options.fixed = [num];
      
      else if( typeof this.options.fixed == "string" || typeof this.options.fixed == "object") 
        if(this.options.fixed.constructor != Array) 
          var selec = this.options.fixed;
          var temparr = [];
          var temp = $(this.element[0]).find(selec);
          var x = this;


          temp.each(function() 
            var i;
            for(i=0;i<x.items.length && x.items[i].item.get(0) != this;++i) 
            if(i<x.items.length) temparr.push(i);
          );
          this.options.fixed = temparr;
        
         


      //Let's determine if the items are being displayed horizontally
      this.floating = this.items.length ? o.axis === 'x' || (/left|right/).test(this.items[0].item.css('float')) || (/inline|table-cell/).test(this.items[0].item.css('display')) : false;

      //Let's determine the parent's offset
      this.offset = this.element.offset();

      //Initialize mouse events for interaction
      $.ui.sortable.prototype._mouseInit.apply(this,arguments);
    ,

    _mouseCapture: function( event )  

      this._fixPrev = this._returnItems();
      return $.ui.sortable.prototype._mouseCapture.apply(this,arguments);
    ,

    _mouseStart: function( event )  

      for(var i=0;i<this.options.fixed.length;++i) 
        var num = this.options.fixed[i];
        var elem = this.items[num];
        if(event.target == elem.item.get(0)) return false;
      

      return $.ui.sortable.prototype._mouseStart.apply(this,arguments);
    ,

    _rearrange: function(event, i, a, hardRefresh) 

      a ? a[0].appendChild(this.placeholder[0]) : 
      i.item[0].parentNode.insertBefore(this.placeholder[0], (this.direction == 'down' ? i.item[0] : i.item[0].nextSibling));

      this._refix(i);



      //Various things done here to improve the performance:
      // 1. we create a setTimeout, that calls refreshPositions
      // 2. on the instance, we have a counter variable, that get's higher after every append
      // 3. on the local scope, we copy the counter variable, and check in the timeout, if it's still the same
      // 4. this lets only the last addition to the timeout stack through



      this.counter = this.counter ? ++this.counter : 1;
      var self = this, counter = this.counter;


      window.setTimeout(function() 
        if(counter == self.counter) self.refreshPositions(!hardRefresh); //Precompute after each DOM insertion, NOT on mousemove
      ,0);

    ,

    _refix: function(a) 
      var prev = this._fixPrev;
      var curr = this._returnItems();

      var Fixcodes = this.options.fixed;

      var NoFixed = [];
      var Fixed = [];
      var Mixed = []
      var post = [];


      for(var i=0;i<Fixcodes.length;++i) 
        var fix_index = Fixcodes[i];
        var fix_item  = prev[fix_index];
        var j = 0;

        for(j=0;j<curr.length && curr[j].item.get(0) != fix_item.item.get(0);++j) 

        curr.splice(j,1);

        Fixed.push(fix_item);
      

      for(var i=0;i<curr.length;++i) 
        if(curr[i].item.get(0) != this.currentItem.get(0)) 
          NoFixed.push(curr[i]);
        
      

      var fix_count = 0;
      var nofix_count = 0;

      for(var i=0;i<Fixed.length + NoFixed.length;++i) 
        if(Fixcodes.indexOf(i) >= 0) 
          Mixed.push(Fixed[fix_count++]);
        
        else 
          Mixed.push(NoFixed[nofix_count++]);
        
      

      var parent = this.currentItem.get(0).parentNode;    
      var allchild = parent.children;

      for(var i=0;i<Mixed.length;++i) 
        parent.removeChild(Mixed[i].item.get(0));
        parent.appendChild(Mixed[i].item.get(0));
      
    ,

    _returnItems: function(event) 

      this.containers = [this];
      var items = [];
      var self = this;
      var queries = [[$.isFunction(this.options.items) ? this.options.items.call(this.element[0], event,  item: this.currentItem ) : $(this.options.items, this.element), this]];
      var connectWith = $.ui.sortable.prototype._connectWith.apply;

      if(connectWith) 
        for (var i = connectWith.length - 1; i >= 0; i--)
          var cur = $(connectWith[i]);
          for (var j = cur.length - 1; j >= 0; j--)
            var inst = $.data(cur[j], 'sortable');
            if(inst && inst != this && !inst.options.disabled) 
              queries.push([$.isFunction(inst.options.items) ? inst.options.items.call(inst.element[0], event,  item: this.currentItem ) : $(inst.options.items, inst.element), inst]);
              this.containers.push(inst);
            
          ;
        ;
      

      for (var i = queries.length - 1; i >= 0; i--) 
        var targetData = queries[i][1];
        var _queries = queries[i][0];

        for (var j=0, queriesLength = _queries.length; j < queriesLength; j++) 
          var item = $(_queries[j]);

          item.data('sortable-item', targetData); // Data for target checking (mouse manager)

          items.push(
            item: item,
            instance: targetData,
            width: 0, height: 0,
            left: 0, top: 0
          );
        ;
      ;

      return items;
    ,


    value: function(input) 
        //console.log("test");
        $.ui.sortable.prototype.value.apply(this,arguments);
    
);

)(jQuery);

为了以防万一

依赖

https://ajax.googleapis.com/ajax/libs/jquery/1.6.1/jquery.min.js https://ajax.googleapis.com/ajax/libs/jqueryui/1.8.13/jquery-ui.min.js

脚本

function randomColor()  //for a little fun ;)
   var r = (Math.floor(Math.random()*256));
   var g = (Math.floor(Math.random()*256));
   var b = (Math.floor(Math.random()*256));

   return "#" + r.toString(16) + g.toString(16) + b.toString(16)


$(function() 
    $("#sortable1").fixedsortable(
        fixed: "> .static", //you can use css selector
        sort: function()   //you can add events as well, without getting confused. for example:
            $(".static").css("background",randomColor())  //change the fixed items background
        ,
        change: function(event,ui) 
            $(ui.item[0]).css("border","2px solid "+randomColor())  //change the captured border color
        ,
        stop: function(event,ui) 
            $(ui.item[0]).css("border","2px solid #777"); //change the back the css modifications
            $("#sortable1 > li.static").css("background","#aaa");
        
    );

    $("#sortable2").fixedsortable(  //you can use jQuery object as selector
        fixed: $("li[foo]").css("background","red")
    );

    $("#sortable3").fixedsortable(
        fixed: [2,4], //you can use array of zero base indexes as selector
        update: function(event, ui) 
            alert($(this).fixedsortable('toArray'))   //the fixedsortable('toArray') also works
        
    )

    $("#sortable4").fixedsortable(
        fixed: 5  //you can fix a single item with a simple integer
    )
);

HTML

 <body>
    <div style="width:120px;float:left;">
    <ul id="sortable1">
        <li><a href="#">oranges</a></li>
        <li class="static"><a href="#">apples</a></li>
        <li><a href="#">bananas</a></li>
        <li><a href="#">pineapples</a></li>
        <li><a href="#">grapes</a></li>
        <li class="static"><a href="#">pears</a></li>
        <li><a href="#">mango</a></li>
    </ul>

    <ul id="sortable2">
        <li>bananas</li>
        <li foo="asd">oranges</li>
        <li foo="dsa">apples</li>
        <li>pineapples</li>
        <li>grapes</li>
        <li>pears</li>
        <li>mango</li>
    </ul>
    </div>
    <div style="width:120px;float:left;">
    <ul id="sortable3">
        <li id="fru_1">bananas</li>
        <li id="fru_2">oranges</li>
        <li id="fru_3" style="background:#f4f">apples</li>
        <li id="fru_4">pineapples</li>
        <li id="fru_5" style="background:#faaba9">grapes</li>
        <li id="fru_6">pears</li>
        <li id="fru_7">mango</li>
    </ul>


    <ul id="sortable4">
        <li>bananas</li>
        <li>oranges</li>
        <li>apples</li>
        <li>pineapples</li>
        <li>grapes</li>
        <li style="background:#dada00">pears</li>
        <li>mango</li>
    </ul>
   </div>
</body>

CSS

ul margin:10px;
ul#sortable1 > li, ul#sortable2 > li, ul#sortable3 > li, ul#sortable4 > li 
    display:block;
    width:100px;
    height:15px;
    padding: 3px;
    background: #aaa;
    border: 2px solid #777;
    margin: 1px;

ul#sortable1 > li.static 
    opacity:0.5;

【讨论】:

【参考方案7】:

也许这会对某人有所帮助:使用“禁用”和“启用”方法。例子 HTML:

<ul class="sortable">
  <li>You can move me</li>
  <li data-state="lifeless">You can't move me.</li>
</ul>

脚本:

$('#sortable').sortable();
$('#sortable').mousedown(function() 
  if($(this).data('state')=='lifeless') $('#sortable').sortable('disable');
  else $('#sortable').sortable('enable');
);

此处的示例:https://jsfiddle.net/ozsvar/0ggqtva5/2/

【讨论】:

几乎但不是一个真正可用的答案,因为使用非禁用项目仍然可以重新排序......【参考方案8】:

连接的可排序项和固定项

当我们有几个连接的可排序对象时,我遇到了这个问题。 @sarunast 和 @DarthJDG 建议的代码在将项目从一个列表拖到另一个列表时存在错误行为。 因此,我对其进行了一些修改,现在您可以从不同的列表中拖动项目,并在两个列表中保存位置。

javascript:

let connected = '.soratble';
let fixed = '.static';
let newParentContainer;

//wrap the code suggested by @sarunast and @DarthJDG into the function
//code was modified a little
function sortingAroundFixedPositions(container) 
  let sortable = $(container);
  let statics = $(fixed, container).detach();
  let tagName = statics.prop('tagName');
  let helper = $('<' + tagName + '/>').prependTo(container);
  statics.each(function() 
    let target = this.dataset.pos;
    let targetPosition = $(tagName, sortable).eq(target);
    if (targetPosition.length === 0) 
      targetPosition = $(tagName, sortable).eq(target - 1)
    
    $(this).insertAfter(targetPosition);
  );
  helper.remove();


$('ul').sortable(
  connectWith: connected,
  cancel: fixed,
  start: function() 
    $(fixed, connected).each(function() 
      this.dataset.pos = $(this).index();
    );
  ,
  change: function(e, ui) 
    sortingAroundFixedPositions(this);
    if (ui.sender) 
      newParentContainer = this;
    
    if (newParentContainer) 
      sortingAroundFixedPositions(newParentContainer);
    
  ,
  update: function(e, ui) 
    newParentContainer = undefined;
  
);

演示:http://plnkr.co/edit/blmv4ZjaWJFcjvO2zQH0

【讨论】:

【参考方案9】:

只需使用“包含/排除”项目选择器。 这是链接:https://jqueryui.com/sortable/#items

【讨论】:

【参考方案10】:

有一个更好的方法来解决这个问题。 您需要使用网格而不是列表,然后您可以通过使用 css 声明元素应在何处排列来固定元素的位置:

.fixed-element 
   grid-column-start: 1;
   grid-column-end: 1;
   grid-row-start: 1;
   grid-row-end: 1;

【讨论】:

以上是关于JQuery 可排序列表和固定/锁定项目的主要内容,如果未能解决你的问题,请参考以下文章

jQuery:可拖动连接到可排序。可拖动项目与可排序列表具有不同的 DOM

如何在 jQuery UI 中将多个可排序列表相互连接?

使用 jQuery UI 可排序插件添加项目后刷新列表

jQuery 可排序和 AJAX 问题。 (当列表来自 AJAX 时无法排序)

如何实现两个可排序的剑道:一个带有固定元素(调色板对象),另一个带有每个元素的副本

为 jquery ui 可排序列表中的每个项目添加一个删除按钮