JavaScript 拖放效果系列四——获得元素的位置

原文来自:[url]http://www.cainiao8.com/web/js_examples/21_tuozhuai4.html[/url]


在第一节我们实现了基本的拖拽效果的时候,window.onload事件的响应函数定义如下:

window.onload = function(){
test = document.getElementById("test");
test.onmousedown = down;
test.onmousemove = move;
document.onmouseup = up;

//下面三行被标出。本页中颜色无法显示
test.style.position = "relative";
test.style.top = "0px";
test.style.left = "0px";
}


注意重点标出的三行代码,我们给元素设置了relative的position属性,这是由于元素只有定义了position属性,它的top和left属性才会有效,才能进一步地制作拖拽的效果。因此给元素假定定位为relative实在是迫于无奈,只能牺牲一些灵活性(当然也可以指定为 absolute,要根据实际情况)。

之后我们又设定元素的top和left属性都为0px,这是为了方便后面的代码获得这两个CSS属性,简化了代码。但是这同时给使用这段代码的文档提了一个要求——要拖拽的元素必须设置top和left属性为0。这显然是一个不太合理的要求。
解决问题

我们在上一个例子中已经看到如何跨浏览器地利用JavaScript获得元素的CSS属性了。现在就来稍稍修改上一个例子中的函数来,让它服务于我们的拖拽代码:

<script type="text/javascript">
……
function dragInit(node){
if(node.className == "drag"){
……
//仍然要求元素是relative定位
node.style.position = "relative";
node.dragging = false;
}
var children = node.childNodes;
for(var i = 0;i < children.length; i++){
dragInit(children[i]);
}
}
window.onload = function(){
dragInit(document);
document.onmouseup = docUp;
}
function down(event)
{
event = event || window.event;
dragElement = this;
mouseX = parseInt(event.clientX);
mouseY = parseInt(event.clientY);
objY = parseInt(getNodeStyle(dragElement,"top"));
objX = parseInt(getNodeStyle(dragElement,"left"));
//IE不返回未设置的CSS属性
if(!objY)objY=0;
if(!objX)objX=0;
this.style.zIndex = max++;
}
……
function getNodeStyle(node,styleName){
var realStyle = null;
if(node.currentStyle){
realStyle = node.currentStyle[styleNmae];
}else if(window.getComputedStyle){
realStyle = window.getComputedStyle(node,null)[styleName];
}
return realStyle;
}


可以看到,我们使用getNodeStyle函数来获得元素的CSS属性值,这样我们的代码就可以适用于事先设置了top和left定位值的元素了。我做了一个测试页面,给两个可拖拽的div分别设置了如下的CSS规则:

<style type="text/css">
.drag{border:1px solid; width:400px; background:#CCCCCC;}
#test1{ top:20px;}
#test2{ left:40px;}
</style>


点击进入[url=http://www.cainiao8.com/web/js_examples/tuozhuai_ex4.html]测试页面[/url]。这样,我们的拖拽代码又改进了一小步。
JavaScript拖拽系列

1. JavaScript拖拽
2. JavaScript拖拽2——多元素、分离JS
3. JavaScript拖拽3——解决快速拖拽的问题
4. JavaScript拖拽4——获得元素的位置
5. JavaScript拖拽5——性能优化
6. JavaScript拖拽6——修复错误
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在拖拽的过程中,我们经常会遇到快速拖拽的问题,也就是鼠标移动速度太快,导致拖拽元素跟不上鼠标的移动速度,造成拖拽效果的不流畅。解决这个问题的方法有很多种,下面介绍两种常见的方法。 #### 方法一:使用 requestAnimationFrame requestAnimationFrame 是一个浏览器提供的 API,它可以让浏览器在下一次重绘之前执行指定的函数,这样可以保证函数执行的频率跟浏览器的重绘频率一致,避免了浏览器过度绘制和卡顿的问题。我们可以利用 requestAnimationFrame 来优化拖拽的效果。 具体实现方法如下: ```javascript let startX, startY, currentX, currentY; let element = document.getElementById('drag'); element.addEventListener('mousedown', function (event) { startX = event.clientX; startY = event.clientY; document.addEventListener('mousemove', onMouseMove); document.addEventListener('mouseup', onMouseUp); }); function onMouseMove(event) { currentX = event.clientX - startX; currentY = event.clientY - startY; requestAnimationFrame(updatePosition); } function updatePosition() { element.style.transform = `translate(${currentX}px, ${currentY}px)`; } function onMouseUp() { document.removeEventListener('mousemove', onMouseMove); document.removeEventListener('mouseup', onMouseUp); requestAnimationFrame(() => { element.style.transform = `translate(${currentX}px, ${currentY}px)`; currentX = 0; currentY = 0; }); } ``` 在 onMouseMove 函数中,我们使用 requestAnimationFrame 来更新元素位置,这样可以保证元素位置更新频率跟浏览器的重绘频率一致,从而避免了拖拽效果的不流畅问题。在 onMouseUp 函数中,我们也使用 requestAnimationFrame 来更新元素位置,并且将 currentX 和 currentY 重置为 0,这样可以避免多次拖拽时元素位置的累加。 #### 方法二:使用 CSS3 动画 除了使用 requestAnimationFrame,我们还可以使用 CSS3 动画来优化拖拽效果。具体实现方法如下: ```javascript let startX, startY, currentX, currentY; let element = document.getElementById('drag'); element.addEventListener('mousedown', function (event) { startX = event.clientX; startY = event.clientY; document.addEventListener('mousemove', onMouseMove); document.addEventListener('mouseup', onMouseUp); }); function onMouseMove(event) { currentX = event.clientX - startX; currentY = event.clientY - startY; element.style.transform = `translate(${currentX}px, ${currentY}px)`; } function onMouseUp() { document.removeEventListener('mousemove', onMouseMove); document.removeEventListener('mouseup', onMouseUp); element.style.transition = 'transform 0.2s ease-out'; element.style.transform = `translate(${currentX}px, ${currentY}px)`; setTimeout(() => { element.style.transition = ''; currentX = 0; currentY = 0; element.style.transform = ''; }, 200); } ``` 在 onMouseUp 函数中,我们首先给元素添加了一个过渡动画,这样可以让拖拽效果更加流畅。然后使用 setTimeout 函数来清除过渡动画,并将 currentX 和 currentY 重置为 0,最后将元素的 transform 属性重置为空。这样可以让元素回到原来的位置,同时避免了多次拖拽时元素位置的累加。 需要注意的是,这种方法需要我们手动设置过渡动画的时间,并且需要等待过渡动画执行完毕之后才能重置元素位置,因此需要在 setTimeout 函数中设置一个适当的等待时间。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值