gpt4 book ai didi

javascript - JQuery Draggable with Touch Punch 停止垂直滚动

转载 作者:行者123 更新时间:2023-11-30 09:41:48 34 4
gpt4 key购买 nike

我正在使用 JQuery Draggable 函数和 Touch Punch 来生成可以通过单击和拖动来滚动的水平 slider 列表。它适用于触摸和点击设备。我面临的问题是,如果我尝试在触摸设备中向上或向下滚动,它不起作用。

我查看了 SO,发现从 TouchPunch 中删除“event.preventDefault”允许垂直滚动,这个修复的问题是,它只适用于某些设备,而不是所有设备。

我想知道是否有人有任何其他解决方案或替代方法来生成适用于触摸和点击事件的相同水平 slider 。

这里是示例代码(JQuery Draggable):

$(function() {
var slides = $('#list1 ul').children().length;
var slideWidth = $('#list1').width();
var min = 0;
var max = -((slides - 1) * slideWidth);

$("#list1 ul").width(slides * slideWidth).draggable({
axis: 'x',
drag: function(event, ui) {
if (ui.position.left > min) ui.position.left = min;
if (ui.position.left < max) ui.position.left = max;
}
});

$("#list2 ul").width(slides * slideWidth).draggable({
axis: 'x',
drag: function(event, ui) {
if (ui.position.left > min) ui.position.left = min;
if (ui.position.left < max) ui.position.left = max;
}
});


});
#list1 {
position: relative;
height: 16em;
width: 100%;
text-align: middle;
white-space: nowrap;
overflow-x: hidden;
overflow-y: hidden;
}

#list1 .floating-box {
margin: auto;
display: inline-block;
width: 15em;
height: 13.5em;
margin: 0.1em;
border: 0.2em solid black;
background-color: white;
}

#list2 {
position: relative;
height: 16em;
width: 100%;
text-align: middle;
white-space: nowrap;
overflow-x: hidden;
overflow-y: hidden;
}

#lis2 .floating-box {
margin: auto;
display: inline-block;
width: 15em;
height: 13.5em;
margin: 0.1em;
border: 0.2em solid black;
background-color: white;
}
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
<div id="list1">
<ul>
<p>One</p>
<div class="floating-box">Floating box</div>
<div class="floating-box">Floating box</div>
<div class="floating-box">Floating box</div>
</ul>
</div>


<div id="list2">
<ul>
<p>Two</p>
<div class="floating-box">Floating box</div>
<div class="floating-box">Floating box</div>
<div class="floating-box">Floating box</div>
</ul>
</div>

如果我触摸 list1 或 list2 div 并尝试向上或向下滚动,它无法识别移动。任何帮助或指导将不胜感激。

最佳答案

编辑
基于根据 touchemovepageY 属性确定滑动方向的想法。

我认为从长远来看,避免烦人的双击是个好主意(请参阅下面的第一个答案)
还是有妥协的地方,但要合理得多。

我尝试了很多东西...我得到的最好结果是当我放弃在 touchmove 上模拟滚动时。

下面是我现在使用三个触摸事件中的每一个的方法:

  • touchstart 获取初始变量,例如:window scrollTop 和 pageY。

  • touchmove确定滑动方向并获取最后一页Y。

  • touchend 计算页面是否应该滚动到。
    为了可爱起见,我将该结果值放在 .animate() 中。
    我惊喜地发现它非常平滑地补偿了页面仅在 touchend 上滚动的事实。
    我认为很少有用户会注意到它;)。

由于“Touch-Punch”默认适用于水平滑动,因此“妥协”仅影响垂直滚动。



这是代码:
还有一个live link在支持触控的设备上试用。

$(function() {
var slides = $('#list1 ul').children().length;
var slideWidth = $('#list1').width();
var min = 0;
var max = -((slides - 1) * slideWidth);

$(".draggable").width(slides * slideWidth).draggable({
axis: 'x',
drag: function(event, ui) {
if (ui.position.left > min) ui.position.left = min;
if (ui.position.left < max) ui.position.left = max;
}
});

var startTouchX=0;
var startTouchY=0;

var actualPosX;
var actualPosY;

var eventCounter=0;
var directionDetermined=false;
var direction;

var thisTouchX;
var thisTouchY;

var lastTouchY;

$(document).on("touchstart", function(e) {
// Actual document position
actualPosX = $(document).scrollLeft();
actualPosY = $(document).scrollTop();

// StartTouches
startTouchX = parseInt(e.originalEvent.touches[0].pageX);
startTouchY = parseInt(e.originalEvent.touches[0].pageY);
});

$(document).on("touchmove", function(e) {

// Arbitrary considering ONLY the fourth event...
// Touchmove fires way too many times!
// We only need to determine the main direction ONCE.
// This prevents an "s" swipe from messing this code.
eventCounter++;
if(eventCounter==4 && !directionDetermined){

thisTouchX = parseInt(e.originalEvent.touches[0].pageX);
thisTouchY = parseInt(e.originalEvent.touches[0].pageY);

if ( (Math.abs(thisTouchX - startTouchX)) / Math.abs(thisTouchY - startTouchY) > 1){ //check swipe direction

// HORIZONTAL
$("#debug").html("HORIZONTAL");
directionDetermined=true;

// NO NEED here. This is re-enabled on touchend, if it has been disabled.
//$(".draggable").draggable('enable');
}
else{

// VERTICAL
$("#debug").html("VERTICAL");
directionDetermined=true;
direction="vertical";

$(".draggable").draggable('disable'); // Disable draggable.
}
}

// Getting all the Y touches...
// The "last" value will be used on touchend.
lastTouchY = parseInt(e.originalEvent.touches[0].pageY);
//$("#debug").html(lastTouchY);
});

$(document).on("touchend", function(e) {
if(direction=="vertical"){
//$("#debug").html(lastTouchY);

var thisMoveY = -parseInt(lastTouchY - startTouchY);
//$("#debug").html(thisMoveY);

var newPosY = (actualPosY + thisMoveY);
//$("#debug").html(actualPosX+ " " +newPosY);

//window.scrollTo(actualPosX, newPosY);
$("html,body").animate({ scrollTop: newPosY },400);
}

// Reset everything for a future swipe.
startTouchX=0;
startTouchY=0;
eventCounter=0;
directionDetermined=false;
direction="";
$("#debug").html("");

// Re-enable draggable.
$(".draggable").draggable('enable');
});
});




第一个答案,使用“双击”切换方向。

首先,Touch Punch 网站声明它基本上是一个 jQuery-UI hack 来处理 一些 实际上 jQuery-UI 未处理的情况...并且有可能找到 Touch Punch 失败的情况。

您的问题已报告给 Touch Punch 开发人员 here .
作为对此的回答 (用我的话来说),他们说这不是真正的错误或问题...
但是在使用相同触摸事件的两个不同“希望”操作上的用法“冲突”。
有时滚动页面,有时拖动元素。

作为解决方案提示,他们发布了 this Fiddle .
它建议找到一种在需要时禁用 draggable 的方法。

但在此解决方案中,可滚动部分可拖动元素中。
这不是你的情况。
而且您几乎所有的移动屏幕空间都用于可拖动元素,因此没有足够的空间来触发它们周围的 draggable("disable")

所以...我有了这个想法,希望对您有所帮助。

如果您找到一种优雅的方式来通知您的用户“双击”会改变移动方向,您会怎样?
在这里,我建议使用一个非常简单的“双箭头”来表示运动方向。
也许您会找到更好的东西。

这肯定是对用户体验的一点妥协,要求他们双击...
但如果您的布局确实需要它,也许它没问题。

所以在这里,我转载了你的initial issue .
这是the fix that I suggest .

我只在 Samsung Galaxy S3 上试过,但应该适用于所有触控设备。

$(function() {
var slides = $('#list1 ul').children().length;
var slideWidth = $('#list1').width();
var min = 0;
var max = -((slides - 1) * slideWidth);

$(".draggable").width(slides * slideWidth).draggable({
axis: 'x',
drag: function(event, ui) {
if (ui.position.left > min) ui.position.left = min;
if (ui.position.left < max) ui.position.left = max;
}
});

// Flag
var draggableEnabled=true;

// Find the doubletap position (to show a nice double arrow)
var tapPosition=[];
$(document).on("touchstart",function(e){
tapPosition[0] = parseInt(e.touches[0].pageX) - $(document).scrollLeft() - ($("#arrows img").width()/2);
tapPosition[1] = parseInt(e.touches[0].pageY) - $(document).scrollTop() - ($("#arrows img").width()/2);
});

Hammer(document).on("doubletap", function() {
//alert("Double tap");
draggableEnabled = !draggableEnabled; // Toggle

if(!draggableEnabled){
$(".draggable").draggable('disable'); // Disables draggable (and touch Punch)
$("#arrows img").css({
"transform":"rotate(90deg)", // Nice vertical double arrow
"top":tapPosition[1],
left:tapPosition[0]
}).fadeIn(600, function(){
$(this).fadeOut(600);
});

}else{
$(".draggable").draggable('enable'); // Enables draggable (and touch Punch)
$("#arrows img").css({
"transform":"rotate(0deg)", // Nice horizontal double arrow
"top":tapPosition[1],
left:tapPosition[0]
}).fadeIn(600, function(){
$(this).fadeOut(600);
});
}
});
});

注意它使用了 Hammer.js ( CDN ) 来检测双击。

还有一些额外的 CSS 用于 double arrow .

#arrows img{
width: 60vw;
height: 60vw;
position: fixed;
top:calc( 50vh - 60vw );
left:calc( 50vh - 60vw );
z-index:1000;
display:none;
}

关于javascript - JQuery Draggable with Touch Punch 停止垂直滚动,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40799506/

34 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com