上篇博客链接:视差滚动的爱情故事
【优化问题 : 解决Chrome下跳动的bug】
在上一篇的爱(diao)情(si)故事里面,demo3在 Chrome下是这样的问题:鼠标滚动视差元素动画生硬,鬼畜跳动,拖动滚动条却没有这样的问题。之前也分析过,是因为Chrome只触发一次 scroll事件导致的(还有个上下跳动的原因,博客后面会再论述),后来IE10也发现也有这个问题。概括一下问题可以论述为:现代浏览器出于“优化” 目的,对scroll事件机制进行特殊处理,限制了鼠标滚动所触发的scroll事件次数。硬件设备鼠标滚动一个齿轮幅度,通常scrollTop就会滚 动了100px左右,其他浏览器会触发十几次,而Chrome和IE10却只触发一次的scroll事件。
http://www.manufacturedessai.it/it/ 这个网站的视差滚动也是没有做优化,视差效果出现问题跟我上个的dome3一样,滚动起来,看着美女下面粗壮的大腿不断地跳啊跳啊,挺惊悚的。(此网站后来换了个烈焰红唇)
【采用mousewheel事件做视差滚动,顺道避规上面的问题】
既然问题出在浏览器的“优化”机制,那么我们只要找出触发浏览器的优化的核心代码,改写即可。或者重新自个写个浏览器,就不会有这样的问题了,而且还能在市场上推广自己的浏览器………….(博主,不就是当个备胎嘛,认真点。)
既然问题出在scroll事件,我们就改用mousewheel事件即可。注意,mousewheel鼠标滚轮事件在浏览器上兼容性比较差,所 以………..大家还是自个写个浏览器吧(喂喂!认真点!)。关于mousewheel的使用和兼容不在本文讨论范围,不清楚的读者自个查(wan)阅 (dan)去吧。
使用mousewheel的的wheelDelta值后,我们就可以忽略掉scrollTop的值,元素要滚动多少内容就完全由我们来决定了,这样可以解决上面的“滚动了几百px却只触发几次的动画事件”的问题。来看个demo。
核心代码如下:
function mouseWheelHandler(e){ var e = e || window.event; //使用wheelDelta的正负来判断鼠标是向上滚动还是向下滚动 //wheelDelta是120的倍数(鼠标下滚是为负),detail是firefox的,3的倍数(鼠标上滚为正)。 var delta = Math.max(-1, Math.min(1, (e.wheelDelta || -e.detail))); circles[0].style.top = getOldTop(circles[0])+delta*10+'px'; circles[1].style.top = getOldTop(circles[1])+delta*20+'px'; circles[2].style.top = getOldTop(circles[2])+delta*50+'px'; }
显然要想使用mousewheel作为视察滚动的触发事件,最好把页面overflow设为hidden。因为出现滚动条后,滚动鼠标,是 mousewheel事件和scroll事件都触发的,但是如果拖动滚动条,那就只会触发scroll事件而不触发mousewheel了。介于没有了滚 动条,可以创造的思路就更多更广了,往往令人震惊的视差滚动都是使用mousewheel的形式来创作的
马里奥赛车:http://www.nintendo.com.au/gamesites/mariokartwii/#home
QQ浏览器7:http://browser.qq.com/index_m.html
愤怒的小鸟星际版(很可惜,这个网站已经被覆盖了)
博客后面还有对上面这两个优秀的视差滚动做了小demo。
【hack的方法解决问题?】
虽然使用mousewheel能够避规问题,但不是解决问题,就像你的女神有了男友,你就对你女神避而远之,你这样对得起自己的狼心狗肺吗?对得起 乡亲叔侄吗?所来,还是像我那样,当一名备胎多好。欢迎加入“寡人要当备胎俱乐部“。好,回归正题,我们还是要直视滚动鼠标触发scroll事件的 bug。我开始的思路是下面这几个:
1.同时绑定scroll事件和mousewheel事件,滚动鼠标就屏蔽scroll事件,只触发mousewheel事件,问题搞定
2.只触发一次scroll事件,那么强制触发多次scroll事件,让触发scroll事件次数回归正常,问题搞定
3.自个写个浏览器,问题搞定
读者自个忽略每个思路最后一句话,单单从思路上看,就知道是hack手法。但没办法,只能试试,介于方法都很麻烦,而且总试不成功,一直处理到深夜,很夜很夜,终于找到一个奇怪的处理手法,下面简称”莫名其妙就解决了问题的hack方法“。自个用Chrome先看demo
hack的手法核心就是:在原来的使用scroll事件的视差滚动,增加一个对mousewheel的事件的”空处理“函数。其实只要有绑定mousewheel事件即可,里面有逻辑处理和没有逻辑逻辑都可以。目前没有找到合理的解析。
手拿着hack手法的我,悄悄走到了IE10姐姐身后,想把IE10也hack掉,那我的大计就完成。
突然IE10姐姐一个华丽的扭头,伴随的还有飘逸飞舞的长发和全新力士香薰沐浴露的香味。
”你就是那个常年做备胎的谭伟X?“
我了个大叉,什么玩意,为什么他会知道我是个备胎,她明明是个浏览器啊,只是萌化娘了一下而已,难道我当备胎的事情我同事我父母我基友都知道吗?先不理了,只要把hackIE10的任务完成了先。
”那个。。。IE10姐姐,你看Chrome阿姨都被我hack了,你能不能。。。。“ 我特意把Chrome姐姐说成阿姨。正当IE10萌娘在做思考的时候,身后突然冒出个小萝莉,不好,是IE6小朋友。
”姐姐,就是他,整天都吐槽我诋毁我hack我,活该当备胎!“IE6小朋友居然热泪盈眶了都,说话还这么突出重点,你两姐妹究竟有多在意我是备胎啊。
各位读者,上面就是我深夜hack浏览器,Chrome成功,却在IE10不成功的故事啦。
【深入分析跳动bug】
虽然hack不成功,但分析了scroll和mousewheel的事件,找到了问题的突破点。先来看,Chrome上的鼠标滚动有动画生硬鬼畜跳动的bug,其实有两个原因:
1.只触发一次scroll只有一帧的动画
2.那一帧的动画,浏览器重绘了两次,导致”动画跳动“
我们来深入分析scroll和mousewheel事件触发的顺序和浏览器渲染过程,以一次鼠标滚动触发浏览器事件和渲染顺序的对比
1.一滚动鼠标,首先触发的是mousewheel事件,之后再到scroll,这时候浏览器没有做渲染,dom的scrollTop不变。
2.之后debug到scroll事件,发现,当刚进入scroll事件处理时候,浏览器就按照默认行为,页面内容上滚,这时dom的scrollTop已发生改变,如图。或者说,浏览器默认行为页面内容上滚,再触发scroll事件
3.最后执行到在scroll事件内我们要进行视察滚动的逻辑,使到dom的scrollTop又发生了一次改变。
上面这三步,基本可以把问题看穿了,一次鼠标滚动,浏览器渲染了两次:
第一次,就是默认浏览器行为,页面上滚。渲染位置一般偏上
第二次,视察滚动逻辑,dom的top改变。渲染位置一般偏下
这样一上一下的进行两次渲染,导致了出现个跳动的bug.
【使用fixed做视差滚动,真正处理问题】
”诶,谭伟X,你要不不搞那个问题先了,看你那天搞到这么晚,不如。。。。“ 女神依旧呼唤着我的全名,还安慰我了有木有,不愧是女神啊,真是心地善良啊~~
”其实,我已经找到方法,你大放心“是的,我找到真正的解决方案。
问题出在一上一下的两次渲染,那我们真正希望的是只做第二次渲染,即是视差滚动逻辑的渲染。但是如果吧页面上滚的渲染屏蔽掉呢?屏蔽掉是不实际的,人要往前行,页面内容也要往上滚。大家还记得上篇博客介绍的视差滚动的原理一吗?使用background-attachment: fixed。这个属性可以让background在页面滚动的时候不跟随内容滚动。这不正正是我们需要的吗?
使用background-attachment: fixed,页面dom结构要发生变化,发生视差滚动的元素都要以background-image的赋值和background-position调整 位置。这是demo的片段html代码,需要注意的是scene类,是需要变得非常的大,方便用backgroundPostion调整视差元素的位置
.scene{ position: absolute; width: 100%; height: 3000px; background-repeat: no-repeat; background-attachment: fixed; } </style> <body> <div id="scene_back" class="scene"></div> <div id="scene_center" class="scene"></div> <div id="scene_front" class="scene"></div>
js基本上只是把原来的top,改为backgroundPostionY而已。
function onScroll(e){ var scrollTop = document.documentElement.scrollTop || document.body.scrollTop; sceneBack.style.backgroundPositionY= 100-scrollTop*.05+'px'; sceneCenter.style.backgroundPositionY = 300-scrollTop*.15+'px'; sceneFront.style.backgroundPositionY = 600-scrollTop*.4+'px'; }
问题算是终于解决了,可是女神给予充分感谢后,很长一段时间没有再联系我了,可能跟男一号过得很好吧,可能刚好没有需要我帮忙的吧,也有可能缘分大概走到这里了吧。现在能做的,就是继续完善视差滚动,等到我们再次相遇,让她觉得依旧值得可靠。
思考一下,视差滚动 一般带来有以下几个问题。
1.页面加载慢,存在大量的效果图片。
2.比一般传统页面卡,有大量动画元素。
3.视差滚动效果太花销,用户注意力没有在内容本身。
4.写视差滚动的程序员都是大帅哥,却依旧单身,当着备胎。
【优化思考一:权衡使用合图】
视差滚动需要很多元素效果图片,由于运动轨迹不同,不能用传统的sprite合图,所以会产生很多额外的http请求。但是这里应该思考,把视差滚动运动轨迹相同的元素合图,例如
<div id="scene_back" class="scene"> <img id="pokemon1" src="./img/001.png"> <img id="pokemon4" src="./img/004.png"> <img id="pokemon7" src="./img/007.png"> </div> <div id="scene_center" class="scene"> <img id="pokemon2" src="./img/002.png"> <img id="pokemon5" src="./img/005.png"> <img id="pokemon8" src="./img/008.png"> </div> <div id="scene_front" class="scene"> <img id="pokemon3" src="./img/003.png"> <img id="pokemon6" src="./img/006.png"> <img id="pokemon9" src="./img/009.png"> </div>
我们特殊把运动规则相同的元素都放到一个div里面,这里面可以把每个scene里的img合图,减少http请求数并且可以统一js操作处理。当然,合图大小如果过大也不好,自个权衡取最优吧。
【优化思考二:添加CSS3缓动动画】
差的视差滚动页面都会有卡顿的感觉,一方面是图片没有加载完成,另外一方面动画执行太生硬。我们在一般的视差滚动里面增加CSS3 transition动画,会有别具一格的效果。
优化思考一和二,请查看这个Demo: v_demo4_sprite_tween.html
【优化思考三:请合理使用视差滚动】
如果视差滚动效果太花销,用户只会顾着拼命的滚动页面,好奇有什么的效果,而不在关注页面内容。这本来是希望用来吸引眼球让用户能关注内容,却本末倒置。首先,视差滚动并不适合一般论坛,门户网站或功能性网站等,通常出现新产品的宣传页面上,就是那种能吸引眼球的宣传产品页面上。现网上视差网站都是宣传某个产品或某个游戏,这也是很多优秀的视差滚动网站地址失效的原因,因为这个产品过了宣传期了。
当然,一般的网站博客都可以使用视差滚动,但建议少部分使用视差效果。优秀的视差滚动网站:http://www.ok-studios.de/home/这 个网站的视差效果只用两层来表现,没有过多的渲染元素。底部层为fixed的background,顶部层为网页内容内容和辅助元素(例如开始右边的那两 只模糊的bird)。没有使用js控制不同层之间的差异滚动速度的动画,所以整个网站非常流畅不卡顿。另外,其实这种优秀视差效果是还需要射鸡师的精心设 计,不,更确切的说,任何一个优秀的视差滚动都需要一名优秀的射鸡师。所以如何泡到一名射鸡师妹子,变成这个问题的核心所在。
依旧是一个冰冷幽暗的夜晚,我正认真地观看一部青年爱国片,讲述如何折腾那些可恶日本人,场面相当粗暴,血肉模糊,满满的马赛克,惨叫不断。突然,手机以一种奇怪的方式响起,居然是女神带过来,而且是最新版手机QQ的视频通话,当时神经错乱地按下立刻就接通了。
”嘿,谭伟X,这么晚在干嘛呢!“ 女神依旧一字不留的呼唤着我的全名,依旧亲切。
”嘿,好久不见啦,,,,哈哈,,,,额,,,“ 我立刻调整手机拍摄到安全位置。
”咦,刚才好像什么画面的略过,声音有点怎么这么奇怪,,,哦,你居然还在看爸爸去那里啊?“
手Q视频通话你能不能不要这么流畅啊,声音传播也太好了,以前我用微信视频通话都一卡一卡的,你手Q体验这么棒想干嘛,想毁掉哥的一生啊。还好机智的我立刻打开v.qq.com随便点了个视频播放。
”我们来聊聊天吧,我心情不好,好久没联系你了“
生活总是这样,坚持的,终会有结果。
我迫不及待要展示我对视差滚动的优化思考,女神惊讶之余称赞若声。我连忙再展示几个有趣的视差滚动效果
【有趣的视差滚动效果实现方式二:路径变动】
视差滚动一般都是纵向的,去掉惯性思维,我们可以让自己网站编程横向滚动的,当然也有打斜移动的等等,这些都统称归类为路径变动的视差滚动吧,其中最具代表性的是马里奥赛车:http://www.nintendo.com.au/gamesites/mariokartwii/#home,页面内容根据赛道的方向移动,非常的酷。
由于是使用了mousewheel,可以不是用backgroundPosition的方式写视差而用left,top属性。核心代码如下,以art1_oldLeft为整个视差运动的判断标准,在某个范围如何”拐弯“。
<div class="container"> <article id="article1"> <h1>阿姆施特朗回旋加速喷气式阿姆施特朗炮</h1> <div class="content">。。。。。。。内容。。。。。。</div> <img src="./img/somepic1.jpg" width="400px" height="250px"> <div class="content">。。。。。。。内容。。。。。。</div> <div class="roadbg1"></div><!-- 注意我道路的dom放到了article里面--> <div class="roadangle"></div> </article> <article id="article2"> <h1>阿姆施特朗回旋加速喷气式阿姆施特朗炮</h1> <div class="content">。。。。。。。内容。。。。。。</div> <img src="./img/somepic1.jpg" width="400px" height="250px"> <div class="content">。。。。。。。内容。。。。。。</div> <div class="roadbg2"></div> </article> </div>
function mouseWheelHandler(e){ var e = e || window.event; //使用wheelDelta的正负来判断鼠标是向上滚动还是向下滚动 //wheelDelta是120的倍数(鼠标下滚是为负),detail是firefox的,3的倍数(鼠标下滚为正)。 var delta = Math.max(-1, Math.min(1, (e.wheelDelta || -e.detail))); var art1_oldLeft = getOldStyle(article1,'left'), art1_oldTop = getOldStyle(article1,'top'), art2_oldLeft = getOldStyle(article2,'left'), art2_oldTop = getOldStyle(article2,'top'); var newTop = 0,range = 100 ; if(art1_oldLeft < 50 && art1_oldLeft > -560){ newTop = art1_oldTop + delta*50; range = 40; }else if(art1_oldLeft <= -560){ newTop = -700; }else{ newTop = 0; } article1.style.top = newTop+'px'; article1.style.left = art1_oldLeft + delta*range + 'px'; article2.style.top = newTop + 729 +'px'; article2.style.left = art2_oldLeft + delta*range + 'px'; }
值得注意的是:你debug马里奥赛车的网站,会发现,每滚一次,整个网站的所有的运动元素的dom的left,top都会改变,对于性能要求高的 我们,看到可能都会冒汗三尺。虽然整站运动流程,非常棒,但我写demo的时候还是把统一运动轨迹的dom放到一个div里面,例如demo中的”道路 “都是以absolute的形式嵌在article里面。js操作也只操作article。
当我自信满满向女神展现我的demo成果的时候,女神态度视乎平平,心不在焉的感觉。不行,女神肯定在怀疑我的实力。
【有趣的视差滚动效果实现方式三:QQ浏览器】
QQ浏览器的官网一直是视差滚动界的佼佼者,http://browser.qq.com/index_m.html,看多少次都非常酷。其实本质跟上面的”路径变动“原理差不多,只是单纯的横向运动加上视差元素被旋转过了而已。先看个挫版的Demo
核心代码如下:(js就不贴出来,自个在Demo查看吧)
.rotateBlock{ position: absolute; left: 300px; top: -2000px; -webkit-transform: rotate(45deg); -webkit-transform-origin: left top; } .block{ width: 2000px; height: 3000px; position: absolute; line-height: 3000px; font-size: 200px; } </style> <body> <div class="container"> <div id="rotateBlock" class="rotateBlock"> <div id="green_b" class="block green_b"> Green <span class="sub_head"> Happiness is under the tree that year</span> </div> <div id="orange_b" class="block orange_b"> Red <span class="sub_head"> Youth is like a fire </span> </div> <div id="blue_b" class="block blue_b"> Blue <span class="sub_head">Life is like a boat </span> </div> </div>
class为block就是我们的视差元素,他的父类为rotateBlock,进行了transform旋转,视差元素扩大一定程度的长宽就能看出效果。
【有趣的视差滚动效果实现方式三:QQ浏览器(带响应式)】
这里可能还有个比较有趣的地方,如何让自己的视差滚动带有响应式设计。上面Demo你拉伸页面是不能让内容跟着浏览器变化而改动页面中心。这里的做法:以一个小宽高的div(可为0px)居中在整个页面上,然后比绝对定位的方式把视差元素填到这个div里面。那么页面的长宽就不会影响到页面内容居中的问题了。
html,body{ overflow: hidden; position: relative; margin: 0; width: 100%; height: 100%; } .center_location{ width: 10px; height: 10px; margin: 0; padding: 0; position: absolute; left: 50%; top: 50%; overflow: visible; /* background: red;*/ } .rotateBlock{ position: absolute; -webkit-transform: rotate(45deg); -webkit-transform-origin: left top; } .block{ width: 1500px; height: 2000px; margin-top: -1000px; position: absolute; line-height: 2000px; font-size: 150px; font-family: '微软雅黑'; color: rgba(255,255,255,.5); -webkit-transition:left .5s; } </style> <body> <div class="center_location"> <div id="rotateBlock" class="rotateBlock"> <div id="green_b" class="block green_b">Green for Youth</div> <div id="orange_b" class="block orange_b">Red for Sunshine</div> <div id="blue_b" class="block blue_b">Blue for Life</div> </div> </div>
”额~~~非常棒,不然这样吧,我先去洗澡,等下再聊“ 看完demo,女神略带羞涩的作这样的回答。
”恩,好吧“
机智的我当然知道女神的意思,说出”我先去洗澡“这样的结束性交流语句,我只能盖上电脑,默默离开。
。。。。。。。。。。。。。。。。。。。
。。。。。。。。。。。
哎~~,女神明明这么晚视频通话我出来吃饭,还说在餐厅内展示demo不容易理解,说要上酒店展示。
哎~~,明明在酒店上,我很认真讲解了我demo和思路,最后女神却以”我要洗澡了“结束交流。
哎~~,以我多年和女神网上聊天的经验表示,她不想聊天了。我只好独自离开了酒店。
博客前的小朋友可能觉得,这次demo素材跟上次不同个层次啊。你想想都2月14号半夜了,情侣都去约会了,剩我在那里写这个烂博客,还有什么心 情。还好,隔壁那对情侣好像吵架了,而且还大打出手,互扇面耳,发出“啪啪啪啪”声音,女的可能被扇痛了,发出惨叫声,真是好开心啊~~~~~哈哈哈啊哈 啊哈哈。
原文:http://www.alloyteam.com/2014/02/optimized-articles-of-parallax-scrolling-love-story/
相关推荐
《视差滚动的爱情故事》这篇文章主要探讨了如何利用视差滚动技术来讲述一个有吸引力的互动故事。视差滚动是一种网页设计技术,它通过不同层的元素以不同的速度滚动,营造出深度感和立体感,从而提升用户体验。在这个...
全屏视差滚动是一种网页设计技术,通过在页面滚动时以不同的速度展示背景图像,创造出深度感和立体感,使用户在浏览时感受到丰富的视觉体验。这种效果常用于网站的首页或产品展示部分,能够吸引用户的注意力并提升...
jQuery Parallax是最基础也最常用的视差滚动插件之一,它基于jQuery库,易于理解和集成。开发者可以通过简单的设置,让背景图像随着滚动条的变化产生视差效果。 3. **Skrollr** Skrollr是一款强大的视差滚动库,...
在Web设计中,视差滚动效果是一种增强用户体验的流行技术,它使得背景元素以较慢的速度移动,相比于前景元素,从而营造出深度感和立体感。"vue-parallax"就是这样一个专门针对Vue.js设计的组件,它能够帮助开发者...
这些库提供了一套现成的API和方法,帮助开发者轻松地创建复杂的视差滚动场景,同时处理浏览器兼容性和性能优化问题。 在实际项目中,我们还需要考虑响应式设计,确保视差滚动效果在不同设备和屏幕尺寸上都能良好...
"jq视差滚动框架特效"的核心原理在于监听用户的滚动行为,当页面滚动时,通过计算滚动的距离,调整不同元素的位置。这种技术通常涉及到CSS3的transform属性和translate3d方法,它们能够高效地实现平移效果,同时提高...
视差滚动的关键在于JavaScript库的运用,例如Skrollr。Skrollr是一个轻量级的库,专门用于实现视差滚动效果。它通过监听用户的滚动行为,然后根据预设的属性值调整页面元素的位置和透明度等样式,实现动态效果。...
视差滚动特效是一种流行且富有创意的网页设计技术,它为用户提供了一种独特而引人入胜的浏览体验。在“视差滚动特效.zip”这个压缩包中,可能包含了一个用于实现这种效果的JavaScript插件或者相关代码示例,文件名为...
视差滚动是一种网页设计技术,它通过创建深度感和立体感来增强用户的浏览体验。当用户滚动页面时,背景图像以较慢的速度移动,而前景元素则以较快的速度移动,这种速度差产生的视觉效果就称为视差滚动。这种技术常...
总的来说,"jQuery + Swiper.js 幻灯片图片视差滚动轮播特效"结合了两种强大的工具,通过精心设计的HTML、CSS和JavaScript代码,创造出引人入胜的交互体验,是现代网页设计中常见的亮点之一。掌握这种技术,将有助于...
视差滚动可以用于创建引人入胜的故事叙述或展示产品的独特方式。 要实现这个效果,我们需要以下步骤: 1. **引入jQuery库**:首先确保在HTML文件中包含了jQuery库,可以通过CDN链接或者本地文件引用。 ```html ...
全屏图片视差滚动切换是一种常见的网页设计技术,它通过创建深度感和动态效果来提升用户的浏览体验。这种效果在现代网页设计中被广泛应用,尤其是用于制作引人入胜的首页或产品展示页面。本资源"全屏图片视差滚动...
标题中的“背景视差滚动插件”是一种网页设计技术,它通过创建背景元素相对于前景元素以不同速度移动的视觉效果,来营造出深度感和立体感。这种效果通常用于提升用户体验,使网页浏览过程更加生动有趣。视差滚动在...
标题“支持视差滚动的ScrollView”所指的是一种定制的`ScrollView`实现,它能够使背景图片在用户滚动时以较慢的速度移动,与子控件的滚动速度形成差异,这种差异性滚动可以增强用户的沉浸感,使界面更具吸引力。...
在实现视差滚动时,需要注意性能优化。过度使用JavaScript进行滚动事件监听和计算可能会造成页面卡顿。可以采用节流或防抖技术限制事件处理函数的执行频率,降低对CPU的影响。同时,合理利用CSS3硬件加速可以显著...
**正文** 视差滚动是一种网页设计技术,它通过创建..."Web开发视差滚动"这个主题涵盖了一系列技术,从基本原理到实现细节,再到优化和设计考量,对于希望提升网页设计水平的开发者来说,是一项值得深入研究的技能。
在IT行业中,视差滚动是一种常见的网页设计技术,它通过创建深度感和动态效果来增强用户的浏览体验。这种效果在游戏领域,特别是像"愤怒的小鸟"这样的2D游戏里非常常见,它使得背景和前景元素以不同的速度移动,营造...
视差滚动可以极大地提升用户体验,特别是在故事叙述型的网页或者产品展示页面中。然而,也需要注意,对于移动设备,由于其性能限制和触摸滚动的特性,实现视差滚动时需要额外的优化,确保在各种设备上都能流畅运行。...