我想知道使用 JavaScript/jQuery 为网站构建 Infinity-Image-Loop-Slider 的最佳(良好可读代码、有害实践代码、可重用性)概念是什么?我不知道如何编写幻灯片放映代码,但什么蓝图符合上述要求。 我的问题的主要焦点是如何排列图片以获得无限循环 slider 的印象。
通过查看来自不同 slider 的代码,我发现了两种解决方案:
-每次显示下一张/上一张图像时更改所有图像的 z-Index。
- 更改图像在 DOM 中的位置。
但是检查和理解别人的代码是非常耗时的——这就是我问这个问题的原因:-)
最佳答案
tl;dr - JSBin 示例:http://jsbin.com/ufoceq/8/
无需太多努力即可创建无限图像 slider 的简单方法如下:为简单起见,假设您有 <n>图像循环滑动,以便在 n 之后th 下一个要可视化的图像是 1 st(反之亦然)。
我们的想法是创建第一张和最后一张图像的克隆
无论您的图片有多少,您最多只需要附加 2 个克隆元素。
再次为简单起见,假设所有图像都是 100px宽,它们被包裹在一个容器中,您可以使用 overflow: hidden 向左/向右移动到一个剪裁的面具中.然后,所有图像都可以很容易地排成一行 display: inline-block和 white-space: nowrap在容器上设置(使用 flexbox 现在更容易)。
对于 n = 4 DOM 结构将是这样的:
offset(px) 0 100 200 300 400 500
images | 4c | 1 | 2 | 3 | 4 | 1c
/* ^^ ^^
[ Clone of the last image ] [ Clone of the 1st image ]
*/
一开始,你的容器会定位在left: -100px (或者也是 margin-left: -100px or even better (for a matter of performance) transform: translateX(-100px) ),所以 slider 可以显示第一张图片。要从一个图像切换到另一个图像,您需要在您之前选择的同一属性上应用 javascript 动画。
当您的 slider 当前位于第 4 个th 图像时,您必须从图像 4 切换至 1c ,所以我们的想法是在动画结束时执行回调,很快将您的 slider 包装器重新定位在真正的 1st 图像偏移处(例如,您将 left: -100px 设置为容器)
这类似于当您的 slider 当前位于第一个st 元素上时:要显示上一张图像,您只需要从图像 1 执行动画。至 4c当动画完成后,您只需移动容器,使 slider 立即定位在第 4 个 图像偏移处(例如,您将 left: -400px 设置为容器)。
您可以在上面的 fiddle 上看到效果:这是最小的 js/jquery我使用的代码(当然代码甚至可以优化,因此元素的宽度不会硬编码到脚本中)
$(function() {
var gallery = $('#gallery ul'),
items = gallery.find('li'),
len = items.length,
current = 1, /* the item we're currently looking */
first = items.filter(':first'),
last = items.filter(':last'),
triggers = $('button');
/* 1. Cloning first and last item */
first.before(last.clone(true));
last.after(first.clone(true));
/* 2. Set button handlers */
triggers.on('click', function() {
var cycle, delta;
if (gallery.is(':not(:animated)')) {
cycle = false;
delta = (this.id === "prev")? -1 : 1;
/* in the example buttons have id "prev" or "next" */
gallery.animate({ left: "+=" + (-100 * delta) }, function() {
current += delta;
/**
* we're cycling the slider when the the value of "current"
* variable (after increment/decrement) is 0 or when it exceeds
* the initial gallery length
*/
cycle = (current === 0 || current > len);
if (cycle) {
/* we switched from image 1 to 4-cloned or
from image 4 to 1-cloned */
current = (current === 0)? len : 1;
gallery.css({left: -100 * current });
}
});
}
});
});
如前所述,这个解决方案并不需要太多的努力和谈论性能,将这种方法与没有循环的普通 slider 进行比较,它只需要在 slider 初始化时进行两次额外的 DOM 插入和一些(非常简单) 管理后向/前向循环的额外逻辑。
这是您同时看到两个元素的另一个示例:在这种情况下,您需要克隆更多元素并对逻辑进行一些简单的更改
https://codepen.io/fcalderan/pen/bGbjZdz
我不知道是否存在更简单或更好的方法,但无论如何希望这对您有所帮助。
注意:如果您还需要有一个响应式画廊,也许this answer也可能有帮助
关于javascript - 无限循环 slider 概念,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15876754/
我脑子里浮现出一些关于一种新编程语言的想法,所以我想我会尝试实现它。一位friend建议我尝试使用Treetop(Rubygem)来创建一个解析器。Treetop的文档很少,我以前从未做过这种事情。我的解析器表现得好像有一个无限循环,但没有堆栈跟踪;事实证明很难追踪到。有人可以指出入门级解析/AST指南的方向吗?我真的需要一些列出规则、常见用法等的东西来使用像Treetop这样的工具。我的语法分析器在GitHub上,以防有人希望帮助我改进它。class{initialize=lambda(name){receiver.name=name}greet=lambda{IO.puts("He
我有多个ActiveRecord子类Item的实例数组,我需要根据最早的事件循环打印。在这种情况下,我需要打印付款和维护日期,如下所示:ItemAmaintenancerequiredin5daysItemBpaymentrequiredin6daysItemApaymentrequiredin7daysItemBmaintenancerequiredin8days我目前有两个查询,用于查找maintenance和payment项目(非排他性查询),并输出如下内容:paymentrequiredin...maintenancerequiredin...有什么方法可以改善上述(丑陋的)代
我收到这个错误:RuntimeError(自动加载常量Apps时检测到循环依赖当我使用多线程时。下面是我的代码。为什么会这样?我尝试多线程的原因是因为我正在编写一个HTML抓取应用程序。对Nokogiri::HTML(open())的调用是一个同步阻塞调用,需要1秒才能返回,我有100,000多个页面要访问,所以我试图运行多个线程来解决这个问题。有更好的方法吗?classToolsController0)app.website=array.join(',')putsapp.websiteelseapp.website="NONE"endapp.saveapps=Apps.order("
我是Ruby的新手,有些闭包逻辑让我感到困惑。考虑这段代码:array=[]foriin(1..5)array[5,5,5,5,5]这对我来说很有意义,因为i被绑定(bind)在循环之外,所以每次循环都会捕获相同的变量。使用每个block可以解决这个问题对我来说也很有意义:array=[](1..5).each{|i|array[1,2,3,4,5]...因为现在每次通过时都单独声明i。但现在我迷路了:为什么我不能通过引入一个中间变量来修复它?array=[]foriin1..5j=iarray[5,5,5,5,5]因为j每次循环都是新的,我认为每次循环都会捕获不同的变量。例如,这绝对
我遇到了一个非常奇怪的问题,我很难解决。在我看来,我有一个与data-remote="true"和data-method="delete"的链接。当我单击该链接时,我可以看到对我的Rails服务器的DELETE请求。返回的JS代码会更改此链接的属性,其中包括href和data-method。再次单击此链接后,我的服务器收到了对新href的请求,但使用的是旧的data-method,即使我已将其从DELETE到POST(它仍然发送一个DELETE请求)。但是,如果我刷新页面,HTML与"new"HTML相同(随返回的JS发生变化),但它实际上发送了正确的请求类型。这就是这个问题令我困惑的
假设我有一个没有特定顺序的随机数数组。假设这些是参加马拉松比赛的人的ID#,他们按照完成的顺序添加到数组中,例如:race1=[8,102,67,58,91,16,27]race2=[51,31,7,15,99,58,22]这是一个简化且有些做作的示例,但我认为它传达了基本思想。现在有几个问题:首先,我如何获得特定条目之前和之后的ID?假设我正在查看运行者58,我想知道谁在他之前和之后完成了比赛。race1,runner58:previousfinisher=67,nextfinisher=91race2,runner58:previousfinisher=99,nextfinishe
我有这个:AccountSummary我想单击该链接,但在使用link_to时出现错误。我试过:bot.click(page.link_with(:href=>/menu_home/))bot.click(page.link_with(:class=>'top_level_active'))bot.click(page.link_with(:href=>/AccountSummary/))我得到的错误是:NoMethodError:nil:NilClass的未定义方法“[]” 最佳答案 那是一个javascript链接。Mechan
defreverse(ary)result=[]forresult[0,0]inaryendresultendassert_equal["baz","bar","foo"],reverse(["foo","bar","baz"])这行得通,我想了解原因。有什么解释吗? 最佳答案 如果我使用each而不是for/in重写它,它看起来像这样:defreverse(ary)result=[]#forresult[0,0]inaryary.eachdo|item|result[0,0]=itemendresultendforainb基本上就
在EloquentRuby(第21页,第一版,第六次打印)一书中,作者(RussOlsen)提倡使用each方法而不是for循环,这与我在其他地方读到的所有内容一致。但是作者还继续说,这样做的一个原因是for循环实际上调用了each方法,所以为什么不直接删掉中间人并使用each?所以我想知道这实际上是如何工作的。为了调查,我确实在github上的Ruby存储库上进行了搜索,但发现很难确定我在哪里/如何看到它的实际效果。重述问题:我如何证明Rubyfor循环实际上是使用each方法实现的? 最佳答案 您可以通过编写一个实现每个的类来展
当我写下面的代码时:x=[1,2,3]x我得到这个输出:[1,2,3,[...]][1,2,3,[...]][1,2,3,[...]]我不应该只得到[1,2,3,[1,2,3]]吗?解释是什么? 最佳答案 这没什么奇怪的。数组的第四个元素就是数组本身,所以当你求第四个元素时,你得到的是数组,当你求第四个元素的第四个元素时,你得到的是数组,当你求第四个元素时,你得到的是数组。第四个元素的第四个元素的第四个元素的元素......你得到了数组。就这么简单。唯一有点不寻常的是Array#to_s检测到这样的递归,而不是进入无限循环,而是返回