如何制作HTML5旋转列表/旋转轮选择器/选择器

时间:2015-10-15 19:51:32

标签: javascript html css html5 animation

我的问题是关于用户ByteHamster给出的答案:How to create JavaScript/HTML5 Spinner List with images?在答案中他/她给出了如何使用html,css和Javascript创建滚动动画的示例。动画允许用户通过单击屏幕上所选数字的上方或下方滚动数字,所选数字显示在动画下方的div中。

我想知道是否可以做类似的事情,但不是让图像上下移动,它可以变成数字轮吗?我的意思是,在上面的例子中,一旦数字达到0,滚动停在一个方向,我想知道是否可以创建一个轮子,用户可以不断地从上到下,或从下到上旋转如果他们愿意这样做。这需要使用3D交互式动画软件吗?

我已经看到了这个问题:HTML5/CSS3 - how to make "endless" rotating background - 360 degrees panorama但我不确定答案是否适用于我的项目,因为它们似乎不是互动的。

由于用户ByteHamster的回答超过3年,我想知道是否有更好的方法来实现html5动画的效果?我是否正确认为示例中的Javascript会使某些设备/浏览器无法启用Javascript? html5方法是否是确保效果适用于大多数设备/浏览器的最佳方法?

1 个答案:

答案 0 :(得分:10)

以下是我从提供的信息中汇总的内容...使用鼠标滚轮,滑动并单击顶部和底部数字。当然要求无限。没有特殊的透视风格(但)我觉得它看起来很不错。仍然可以自然选择。没有使用我在评论中链接到的插件或requestAnimationFrame,但jQuery animate()是一个非常好的工具。该库具有很好的跨浏览器支持(实际上它的强度),它需要的只是一个链接,以便JavaScript能够执行。您可以使用CDN,此版本也适用于IE8-:

<script src="//cdnjs.cloudflare.com/ajax/libs/jquery/1.11.3/jquery.min.js"></script>

为了获得使用鼠标滚轮的最佳跨浏览器支持,此插件包括在内:

<script src="//cdnjs.cloudflare.com/ajax/libs/jquery-mousewheel/3.1.12/jquery.mousewheel.js"></script>

https://plugins.jquery.com/mousewheel/

只是一个基本的父母和样式,每个数字都有跨度,有些是在上升的情况下预先设置。

$(function() {

var gate = $(window),
cog = $('#rotator'),
digit = cog.find('span'),
field = $('#result'),
slot = digit.height(),
base = 1.5*slot,
up, swipe;

if (document.readyState == 'complete') interAction();
else gate.one('load', interAction);

function interAction() {

	field.text(0);

	cog.scrollTop(base).fadeTo(0,1).mousewheel(function(turn, delta) {

		if (isBusy()) return false;

		up = delta > 0;
		nextNumber();

		return false;
	});

	digit.on('touchstart', function(e) {

		var begin = e.originalEvent.touches[0].pageY;

		digit.on('touchmove', function(e) {

			var yaw = e.originalEvent.touches[0].pageY-begin;
			up = yaw < 0;
			swipe = Math.abs(yaw) > 30;
		});

		gate.one('touchend', function() {

			digit.off('touchmove');

			if (swipe && !isBusy()) nextNumber();
		});
	})
	.on('mousedown touchstart', function(e) {

		if (e.which && e.which != 1) return;

		var zest = this, item = $(this).index();

		$(this).one('mouseup touchend', function(e) {

			digit.off('mouseup');

			var quit = e.originalEvent.changedTouches;

			if (quit) var jab = document.elementFromPoint(quit[0].clientX, quit[0].clientY);
			if (swipe || item == 2 || quit && jab != zest || isBusy()) return;

			up = item == 1;
			nextNumber();
		});

		return false;
	})
	.mouseleave(function() {

		digit.off('mouseup');
	});
}

function isBusy() {

	return cog.is(':animated');
}

function nextNumber() {

	var aim = base;
	swipe = false;

	up ? aim += slot : aim -= slot;

	cog.animate({scrollTop: aim}, 250, function() {

		up ? digit.eq(0).appendTo(cog) : digit.eq(9).prependTo(cog);

		digit = cog.find('span');

		cog.scrollTop(base);
		field.text(digit.eq(2).text());
	});
}
});
body {
  background: grey;
}

#ticker {
  width: 150px;
  text-align: center;
  margin: auto;
}

#rotator {
  height: 140px;
  font-family: "Times New Roman";
  font-size: 50px;
  line-height: 70px;
  background-image:
  url(http://ataredo.com/external/image/flip.png),
  url(http://ataredo.com/external/image/flip.png),
  url(http://ataredo.com/external/image/flip.png);
  background-position: 0 0, 50% 50%, 100% 150%;
  background-size: 300% 50%;
  background-repeat: no-repeat;
  margin: 0 0 10px;
  overflow: hidden;
  opacity: 0;
}

#rotator span {
  width: 100%;
  height: 50%;
  display: inline-block;
  cursor: default;
  -webkit-tap-highlight-color: rgba(0,0,0,0);
  -webkit-tap-highlight-color: transparent;
}

#result {
  height: 30px;
  font-size: 20px;
  color: white;
  line-height: 30px;
  letter-spacing: 3px;
  -webkit-box-shadow: 0 0 3px black;
  box-shadow: 0 0 3px black;
}
<script src="//ajax.googleapis.com/ajax/libs/jquery/1.11.1/jquery.min.js"></script>
<script src="//cdnjs.cloudflare.com/ajax/libs/jquery-mousewheel/3.1.12/jquery.mousewheel.js"></script>

<div id="ticker">
 <div id="rotator">
   <span>8</span>
   <span>9</span>
   <span>0</span>
   <span>1</span>
   <span>2</span>
   <span>3</span>
   <span>4</span>
   <span>5</span>
   <span>6</span>
   <span>7</span>
 </div>
 <div id="result"></div>
</div>

非常简单,向上或向下动画滚动位置,然后根据方向追加或预先添加第一个或最后一个数字。可以在此处设置动画的持续时间:

cog.animate({scrollTop: current}, 250, function() {

更新 - 在获得一些新的见解之后,例如touchend事件始终触发原始元素,我已经彻底检查了代码。除此之外,它现在有一个精灵背景,它将与数字本身的大小成比例。还改进了整体逻辑并删除了嵌套的侦听器故障。

此编辑的另一个原因是插入一个允许有多个代码(并预设数字)的演示。由于我甚至超越了这一点(添加了直接响应功能),我认为最好还是留下最小的工作代码:

$(function() {

var gate = $(window),
orb = document,
cog = $('.rotator'),
field = $('#result'),
slot = cog.height()/2,
base = 1.5*slot,
list = [],
niche = [7,7,7],
term = 250, // duration of animation
mass, up = true,
yaw = 'mousemove.ambit touchmove.ambit',
hike = 'mouseup.turf touchend.turf',
part = 'mouseleave.range';

tallyCells();

if (orb.readyState == 'complete') interAction();
else gate.one('load', interAction);

gate.on('mouseleave touchcancel', function(e) {

	!(e.type == 'mouseleave' && e.relatedTarget) && lotRelease();
});

function interAction() {

cog.scrollTop(base).each(function(unit) {

	var pinion = $(this),
	digit = pinion.find('.quota'),
	cipher = Number(niche[unit])%10 || 0;
	list[unit] = digit;
	niche[unit] = 0;
	field.append(0);

	for (var i = 0; i < cipher; i++) nextNumber(pinion, unit, true);

	pinion.mousewheel(function(turn, delta) {

		if (isBusy(pinion)) return false;

		up = delta > 0;
		nextNumber(pinion, unit);

		return false;
	});

	digit.on('mousedown touchstart', function(e) {

		if (e.which && e.which != 1) return;

		var zest = this, ken = {}, item = $(this).index();

		tagPoints(e, ken);

		digit.on(part, wipeSlate).on(hike, function(e) {

			wipeSlate();

			var quit = e.originalEvent.changedTouches;

			if (quit) var jab = orb.elementFromPoint(quit[0].clientX, quit[0].clientY);
			if (item == 2 || quit && jab != zest || isBusy(pinion)) return;

			up = item == 1;
			nextNumber(pinion, unit);
		});

		gate.on(yaw, function(e) {

			hubTrace(e, ken);
		})
		.on(hike, function() {

			lotRelease();

			if (!ken.hit || isBusy(pinion)) return;

			up = ken.way < 0;
			nextNumber(pinion, unit);
		});

		return false;
	});

}).fadeTo(0,1);

function tagPoints(act, bit) {

	var nod = act.originalEvent.touches;
	bit.mark = nod ? nod[0].pageY : act.pageY;
	bit.veer = false;
}

function hubTrace(task, gob) {

	var peg = task.originalEvent.touches,
	fly = peg ? peg[0].pageY : task.pageY;
	gob.way = fly-gob.mark;
	gob.hit = Math.abs(gob.way) > 30;

	if (!gob.veer && gob.hit) {
	gob.veer = true;
	wipeSlate();
	}
}

function wipeSlate() {

	mass.off(part + ' ' + hike);
}

function isBusy(whirl) {

	return whirl.is(':animated');
}

function nextNumber(aim, knob, quick) {

	var intent = base, hook = list[knob];

	up ? intent += slot : intent -= slot;

	if (quick) {
	aim.scrollTop(intent);
	revolveTooth();
	}
	else aim.animate({scrollTop: intent}, term, revolveTooth);

function revolveTooth() {

	up ? hook.eq(0).appendTo(aim) : hook.eq(9).prependTo(aim);

	list[knob] = aim.find('.quota');
	niche[knob] = Number(list[knob].eq(2).text());

	aim.scrollTop(base);
	field.text(niche.join(''));
}
}
}

function lotRelease() {

	gate.off(yaw).add(mass).off(hike);
	mass.off(part);
}

function tallyCells() {

	cog.each(function() {

		for (var i = 0; i < 10; i++) {

		var n; !i ? n = 8 : (i == 1 ? n = 9 : n = i-2);

		$(this).append('<div></div>').find('div').eq(i).text(n).addClass('quota');
		}
	});

	mass = $('.quota');
}
});
body {
  text-align: center;
  background: grey;
}

#ticker, .rotator {
  display: inline-block;
}

.rotator {
  width: 100px;
  height: 140px;
  font-family: "Times New Roman";
  font-size: 50px;
  line-height: 80px;
  background-image:
  url(http://ataredo.com/external/image/flip.png),
  url(http://ataredo.com/external/image/flip.png),
  url(http://ataredo.com/external/image/flip.png);
  background-position: 0 0, 50% 50%, 100% 150%;
  background-size: 300% 50%;
  background-repeat: no-repeat;
  margin: 0 0 10px;
  overflow: hidden;
  opacity: 0;
}

.quota {
  height: 50%;
  cursor: default;
  -webkit-tap-highlight-color: rgba(0,0,0,0);
  -webkit-tap-highlight-color: transparent;
}

#result {
  height: 35px;
  font-size: 20px;
  color: white;
  line-height: 35px;
  letter-spacing: 3px;
  -webkit-box-shadow: 0 0 3px black;
  box-shadow: 0 0 3px black;
}
<script src="//ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
<script src="//cdnjs.cloudflare.com/ajax/libs/jquery-mousewheel/3.1.12/jquery.mousewheel.js"></script>

<div id="ticker">
  <div class="rotator"></div>
  <div class="rotator"></div>
  <div class="rotator"></div>
  <div id="result"></div>
</div>

它将自动填充数字,因此无需编写标记。也响应鼠标拖动。

脚本的最新发展可以在这里找到:

codepen.io/Shikkediel/pen/avVJdG

最终更新 - 使用transition而不是jQuery .animate的3d版本。轮子由围绕x轴的单独旋转元件组成,创建一个基本上无限的十边形,而不需要预先添加或附加元素:

codepen.io/Shikkediel/pen/qpjGyq

齿轮是“可以轻弹的”,使它们以用户给出的速度前进 - 然后在点击时再次停止。与原始演示相比,它们对鼠标滚轮事件的响应速度更快。我遗漏点击事件的两个原因,而不是早期的脚本。浏览器支持也有点受限但很好 - 我已经做了额外的努力使其与IE10 +兼容。