首先,解释一下我个人对前端无伤迁移的理解,即移动端和PC端使用同一套代码,或者说原本在PC端运行得很完美的代码,只要修改少许,就可以在移动端完美运行。

当然,大部分的公司会专门为移动端设计了一套,同时,有为PC端设计了一套,这是最完美的结果。但某些情况下,PC和移动端同时用一套时,就要想办法解决了。

最近刚做完一个项目,项目组要求前端同时支持PC端和平板电脑,UI不太复杂,所以布局问题不大,但交互就有点麻烦了。

众所周知,移动端的click事件有300ms的延迟,所以首要的问题要解决这个。由于之前阅读过Zepto的源码,所以第一时间把要模拟tap事件的想法去掉了,这个实在是太可怕了,我除了要绑定click事件外,我还要另外绑定个tap事件,除此之外,我还要另外绑定个touchstart||touchmove||touchend事件,把click事件禁用掉,或者是判断是否为移动设备,再根据这个来绑定tap或click事件,修改的成本实在太大了。想到某个前端技术群里面有人推荐了fastclick,打开源码看了一下,狂喜。我根本无须修改任何代码,只需要引入然后再添加FastCLick.attach就行了,至此,click事件在移动端300ms的延迟完美解决。

注:其实fastclick的实现原理跟zepto的tap模拟差不多,都是基于touchstart,touchmove还有touchend实现的,但fastclick最终会手动出发click事件,即dispathEvent,而zepto则是触发tap事件,同时fastclick会在touchstart时调用e.preventDefault(),以防止click事件的二次触发。

紧接着,就要面临第二个问题了。在PC端,有鼠标事件,例如,mousedown,mousemove,mouseup,相对应地,移动端有touchstart,touchmove和touchend事件,如何将这两个映射起来呢?如何做到在PC端就绑定mouse事件,在移动端就绑定touch事件呢?由于这个页面是基于jQuery搭建起来的,所以,必须研究一下jQuery的实现。果不其然,我在jQuery.event.add看到了如下代码:

即使不看源码,从注释中我们也可以得知,当special[type]存在时,如果selector存在,即取special.delegateType,否则取special.bindType;若special[type]不存在,取原type,根据这个,我们可以做一些映射了。

var ua = navigator.userAgent;
var isMobileDevice = /Windows Phone|Android|iP(?:ad|hone|od)|BB10/.test(ua); if (isMobileDevice) {
special = jQuery.event.special;
special.mousedown = {'delegateType': 'touchstart', 'bindType': 'touchstart'};
special.mouseup = {'delegateType': 'touchend', 'bindType': 'touchend'};
special.mousemove = {'delegateType': 'touchmove', 'bindType': 'touchmove'};
}

这样的话,当代码是如下时:

$(document).on('mousedown', function() {
console.log('a');
});

其在PC端,绑定的是mousedown事件,在移动端,绑定的是toushstart事件了。

最后,还有一个非常关键的问题,就是坐标的获取。当你在PC端的时候,你可以使用e.pageX来获取,但移动端,你却要使用e.originalEvent.changedTouches[0].pageX来获取,这个必须得解决。

从上面的代码可以得知,事件在jQuery.event.dispath中触发,而在这个方法中

jQuery对调用jQuery.event.fix()来对原生的event做一些兼容性的修改,那么我们可以在jQuery.event.dispath做一些文章。

  var ua = navigator.userAgent, isMobileDevice,
oldDispath, special; oldDispath = jQuery.event.dispatch;
isMobileDevice = /Windows Phone|Android|iP(?:ad|hone|od)|BB10/.test(ua); if (isMobileDevice) {
special = jQuery.event.special;
special.mousedown = {'delegateType': 'touchstart', 'bindType': 'touchstart'};
special.mouseup = {'delegateType': 'touchend', 'bindType': 'touchend'};
special.mousemove = {'delegateType': 'touchmove', 'bindType': 'touchmove'}; jQuery.event.dispatch = function(event) {
var args = Array.prototype.slice.call(arguments, 1), touch; event = jQuery.event.fix(event); //修复touch事件
if (event.originalEvent && event.originalEvent.changedTouches) {
touch = event.originalEvent.changedTouches[0];
event.pageX = touch.pageX;
event.pageY = touch.pageY;
} args.unshift(event); oldDispath.apply(this, args);
} }

至此,你无须再修改其他代码,只需要添加这些补丁,就可以做到无伤迁移了。

基于jQuery的前端如何做到无伤迁移的更多相关文章

  1. 基于JQuery的前端form表单操作

    Jquery的前端表单操作:     jquery提供了良好的方法封装,在一些基本的操作的时候,能节省很多的麻烦,其中,在具体使用时,form表单的数据提交是最频繁也最常见的前后数据交换方式,所以在前 ...

  2. 前端关于 superSlide.js 使用,一款基于jquery的前端控件

    1引用jQuery.js 和 jquery.SuperSlide.js 2 编写HTML ** 以下是默认的HTMl结构,分别是 ".hd" 里面包含ul, ".bd&q ...

  3. 前端基于jquery的UI框架

    正在做的一个项目选择jquery作为前端js核心库.然后就想选一个基于jquery的ui库,然后悲催的事情发生了. 至于为什么使用jquery,一是因为不想为授权费用,而又不想引起可能法律纠纷:另一方 ...

  4. 10款基于jquery的web前端特效及源码下载

    1.jQuery时间轴插件:jQuery Timelinr 这是一款可用于展示历史和计划的时间轴插件,尤其比较适合一些网站展示发展历程.大事件等场景.该插件基于jQuery,可以滑动切换.水平和垂直滚 ...

  5. 8款基于Jquery的WEB前端动画特效

    1.超炫酷的30个jQuery按钮悬停动画 按钮插件是最常见的jQuery插件之一,因为它用途广泛,而且配置起来最为方便.今天我们要分享的是30个超炫酷的jQuery悬停按钮动画,当我们将鼠标滑过按钮 ...

  6. 10款基于jquery的web前端动画特效

    1.jQuery横向手风琴图片切换动画 之前我们为大家分享过很多款基于jQuery和CSS3的手风琴菜单和手风琴焦点图插件,比如CSS3响应式垂直手风琴菜单和jQuery横向手风琴图片展示插件.今天要 ...

  7. 基于jquery的ui选择之路

    选定: 主框架:jqueryUi tree:ztree grid:jqGrid layout:jquery.layout 原由: 还有其他demo,ajax实现等参看连接: 正在做的一个项目选择jqu ...

  8. 构建一个基本的前端自动化开发环境 —— 基于 Gulp 的前端集成解决方案(四)

    通过前面几节的准备工作,对于 npm / node / gulp 应该已经有了基本的认识,本节主要介绍如何构建一个基本的前端自动化开发环境. 下面将逐步构建一个可以自动编译 sass 文件.压缩 ja ...

  9. 基于jQuery的Validate表单验证

    表单验证可以说在前端开发工作中是无处不在的~ 有数据,有登录,有表单, 都需要前端验证~~  而我工作中用到最多的就是基于基于jQuery的Validate表单验证~  就向下面这样~ 因为今天有个朋 ...

随机推荐

  1. AspnetPager放在UpdatePanel中,回到顶部。

    最近在做一个项目时,使用了AspNetPager分页控件进行分页,为了防止点击下一页时搜索条件消失掉,使用了UpdatePanel来进行局部刷新. 由此引发了一个问题,即点击某一页时,页面没有返回到顶 ...

  2. iOS 生成.a文件

    一.新建一个工程,选择Cocoa Touch Static Library. 二. 三. 四. 五. 六. 七. 八. 九. 十. 十一. 十二. 十三.打开终端,输入以下命令将真机和模拟器中的.a合 ...

  3. Sql Server获得每个表的行数

    SELECT o.[name], ddps.[row_count] FROM sys.indexes AS i INNER JOIN sys.objects AS o ON i.[object_id] ...

  4. 在sql server中怎样获得正在执行的Sql查询

    方法1:使用DBCC inputbuffer(spid) 使用SP_WHO获得SPID,然后再执行上面的DBCC command,参见下图 执行一段sql语句 打开另一个query窗口并执行SP_WH ...

  5. html 调用 activeX(c++)

    1.新建MFC ActiveX 2.添加方法 3.找到add函数编写代码 4.在test.idl中找到最后一个uuid 5.编译工程,会自动注册控件 6.html中的代码 <html> & ...

  6. 智能电视TV开发---如何实现程序省电

    对于很多使用智能手机的用户来,很多抱怨手机耗电太快,很多人买手机的时候卖家都是推荐买两块电池,还有如果用户留心的话,在买手机的网页上,卖家会显示播放视频多长时间,听音乐多长时间,待机多长时间,不过看的 ...

  7. android.view.InflateException: Binary XML file line #7: Error inflating class(OOM)

    由于页面含有ImageView引起的内存溢出. 作如下处理:在OnDestroy中 Drawable d = imageView.getDrawable(); if (d != null) d.set ...

  8. ecshop各个文件夹作用

    Images文件夹: 这个文件夹下有子文件夹,在这些子文件夹下面存放着当前网站商品的原始图片和缩略图.这些子文件夹命名有规律的,它们的文件名是它们目录下商品加入 的年月份.也就是说在同一个月份加入的商 ...

  9. Yii系列总结:yii 标签用法

    yii 常用标签:label标签.文本标签.error标签.textarea标签.hidden标签.password标签.url标签.radio标签.file标签.button标签.checkBox标 ...

  10. python之列表、字典的使用

    一.概述:以后你在Linux里面写Python脚本的时候会经常用到Python列表.字典,因为你在以后写脚本的时候,大多数情况下都是对文件进行操作处理,使用字典和列表可以很好的操作文件,得出你想要的结 ...