基于JQuery的列表拖动排序实现代码
要求 拖动排序,从名字就不难想像,就是按住一行数据拖到想要的排序位置,保存新的排序队列。 思路 首先给列表行建立锚点,绑定mousedown和mouseup事件,当鼠标移动到想要插入的位置时,将对象行移动到目标行,然后对其经过的所有行进行排序处理。 思路很简单,但这里面仍然有几个问题要注意 1、移动到什么位置可以视作要插入到目标行的位置。 解决 关于事件 Javascript里鼠标按下和放开事件为onmousedown,onmouseup,JQuery里是mousedown,mouseup,所以,这里使用mousedown和mouseup 首先,要知道界面有多少行,每一行有多高,因为要判断鼠标的移动距离 复制代码 代码如下: var tbodyHeight=setting.frame.outerHeight(); //setting.frame,父对象 var lineNum=$("."+setting.dgLine).length; //setting.dgLine,每一行的classname var lineHeight=Math.ceil(tbodyHeight/lineNum); 之所有要取lineNum(行数),除了计算行高外,还有个目的是要使用index(),通过序列索引值来进行移动行到目标位置 当mousedown事件触发后,就要开始计算鼠标移动的距离,用于判断该行到底要移动到什么位置 复制代码 代码如下: dgid=$(this).attr(setting.id); //移动行的ID,setting.id,是每一行用来标记ID的名称 thisIndex=$("#"+setting.linePre+dgid).index(); //该行的索引,setting.linePre,每一行ID关辍 thisLineTop=$("#"+setting.linePre+dgid).offset().top; //该行的top值 topDistance=thisIndex*lineHeight; //该行距离第一行顶端的距离 downDistance=(lineNum-thisIndex-1)*lineHeight; //该行距离最后一行底端的距离
thisLineTop,主要是用来和鼠标移动位置进行高度计算,然后根据行高、索引值来判断是移动到哪一行了。还有个作用是用来确定是否按在了移动锚点上,如果有值,说明是,那后面的mouseup就是成立的,如果没有值,说明没有按到锚点上,mouseup不执行任何操作。为什么要这样做呢?因为不管在页面的什么位置点鼠标,都会触发mouseup事件,如果没有一个判断,就会不停的执行,那就会产生一些问题。 topDistance和downDistance,用来判断鼠标有没有移出列表的范围,如果移除了,也就是鼠标移动的距离大于topDistance或downDistance,那就可以判断为需要移动到第一行或最后一行。 mousedown事件主要做的,就是这几件事情,当然,为了效果,还可以增加一些东西 复制代码 代码如下: $("#"+setting.linePre+dgid).css('background',setting.lineHighlight); //高亮移动行 var left=e.pageX+20; var top=e.pageY; dg_tips(left,top); //创建一个提示层 $('body').css('cursor','move'); //更改页面的鼠标手势 $("body").disableSelection(); //禁止按下鼠标后移动鼠标时选中页面元素 setting.frame.mousemove(function(e){ //让提示层跟随鼠标移动 $("#dgf").css({"left":e.pageX+setting.tipsOffsetLeft+'px',"top":e.pageY+'px'}); }); 这些的目的,只是让操作起来更有效果,比如高亮行,就是要让用户知道,他们操作的是哪一行。提示层的作用也一样。 关于禁止选中,.disableSelection();这是jQuery_UI里带的方法,如果你有使用jQuery_UI,那可以直接使用它,如果没有使用,可以这样做 复制代码 代码如下: $('body').each(function() { $(this).attr('unselectable','on').css({ '-moz-user-select':'none', '-webkit-user-select':'none', 'user-select':'none' }).each(function() { this.onselectstart = function() { return false; }; }); }); 取消禁止选择 复制代码 代码如下: $('body').each(function() { $(this).attr('unselectable','').css({ '-moz-user-select':'', '-webkit-user-select':'', 'user-select':'' }); }); 考虑到通用性,所以后面的代码里,不会使用.disableSelection(); 好了,下面是mouseup事件。这里mouseup事件是绑定在body上的,因为mouseup如果只是绑定在锚点上,那当鼠标移出锚点的时候,再松开鼠标,会发现,这个mouseup事件不执行了,因为它会认为是别的对象的mouseup。所以,最保险的方法是用$('body').mouseup。这样基本上就不会有问题。 mouseup触发后,首先就要判断thisLineTop是不是有值,防止无意义的事件执行。跟着判断鼠标移动的距离是正还是负,也就是向上移动还是向下移动。 var moveDistance=e.pageY-thisLineTop; 根据不同的方向作不同的处理 复制代码 代码如下: if(moveDistance<0){ if(thisIndex!=0){ moveDistance=Math.abs(moveDistance); //为负数的时候,取一下绝对值 if(moveDistance>lineHeight/2){ //判断移动距离是否超过行高的1/2 if(moveDistance>topDistance){ //如果移动距离大于该行到顶边的距离 focusIndex=0; }else{ focusIndex=thisIndex-Math.ceil(moveDistance/lineHeight); } $("."+setting.dgLine).eq(focusIndex).before($("#"+setting.linePre+dgid));//将该行插入目标位置 } } }else{ if(thisIndex!=lineNum-1){ if(moveDistance>lineHeight/2+lineHeight){ if(moveDistance>downDistance){ focusIndex=lineNum-1; }else{ focusIndex=thisIndex+Math.ceil(moveDistance/lineHeight)-1; } $("."+setting.dgLine).eq(focusIndex).after($("#"+setting.linePre+dgid)); } } } 之所以判断移动距离是否超过行高的1/2,是因为如果只移动一小点,可以视作不移动。在计算目标索引值的时候,使用了Math.ceil,最进位,而当移动距离大于0的时候,取了进位还要-1,因为是向下嘛。 向上移动和向下移动使用了不同的插入方法,before和after,可以试着想一下为什么要使用before和after。 移动完了,还得把按下鼠标时使用的效果给去除掉 复制代码 代码如下: $("#dgf").remove();//移除提示层 $("#"+setting.linePre+dgid).css('background','');//将高亮的行变为普通 dgid='';//将移动行的ID值空 thisLineTop=0;//将移动行的Top值至0 $('body').css('cursor','default');//更改鼠标手势为默认 基本上的情况就是这样,主要问题就是在处理移动和判断在哪里插入的问题上。其它的都非常简单。 下面给出完整的封装程序,包括更新数据部分 复制代码 代码如下: /* * * DragList.js * @author fuweiyi <fuwy@foxmail.com> * */ (function($){ $.fn.DragList=function(setting){ var _setting = { frame : $(this), dgLine : 'DLL', dgButton : 'DLB', id : 'action-id', linePre : 'list_', lineHighlight : '#ffffcc', tipsOpacity : 80, tipsOffsetLeft : 20, tipsOffsetTop : 0, JSONUrl : '', JSONData : {}, maskLoaddingIcon : '', maskBackgroundColor : '#999', maskOpacity : 30, maskColor : '#000', maskLoadIcon:'', }; var setting = $.extend(_setting,setting); var dgid='',thisIndex,thisLineTop=0,topDistance,downDistance; $("."+setting.dgButton).mousedown(function(e){ 使用 复制代码 代码如下: <table cellpadding="5" cellspacing="0" class="listtable" id="listtable"> <thead> <tr> <td class="td50">拖动</td> <td class="td220">名称</td> </tr> </thead> <tbody id="drag_table"> <!--{loop $lists $k $list}--> <tr id="list_{$list['id']}" action-id="{$list['id']}" class="drag_line"> <td><div class="drag_orders" action-id="{$list['id']}"><img src="{SYS_URL}views/admin/images/move.png" alt="" /></div></div></td> <td>这里是一行</td> </tr> <!--{/loop}--> </tbody> </table> <script type="text/javascript"> $("#drag_table").DragList({ dgLine:'drag_line', dgButton:'drag_orders', id:'action-id', linePre:'list_', JSONUrl:'{_ADMIN}?controller=contents&method=focus_form', maskLoadIcon:'{SYS_URL}views/admin/images/loading.gif' }); </script> 参数主要是dgLine,dgButton,id,linePre和JSONUrl,通过看HTML代码,应该不难理解。 关于拖动排序就是这么多了,当然还可以把效果做得更漂亮些,提示更清楚点,有助于用户体验 (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |