1. pageinit & pageshow
JQM的官方手册重点提醒了使用$(document).bind(‘pageinit’)代替$(document).ready()。
但当你需要对某一个页面(page)编写其独享的Javascript脚本时, 选择器应该选择的是该page层, 而不是document, 并使用live()
添加事件处理器。这在ajaxEnable=true的情况下尤为重要。
JS :
$(document).bind(‘pageinit‘, function () { console.log(‘任意一个页面初始化时响应‘); }); $(document).bind(‘pageshow‘, function () { console.log(‘任意页面被显示时响应‘) }); $("#hello").live(‘pageinit‘, function () { console.log(‘只响应id为hello的页面初始化‘); }); $("#hello").live(‘pageshow‘, function () { console.log(‘只响应id为hello的页面被显示‘); });
Html :
id=‘hello‘ data-role=‘page‘>
data-role="content"> href="#world" data-role="button">Next
id=‘world‘ data-role=‘page‘>
data-role="content"> href="#hello" data-role="button">Previous
2. refresh
通过脚本修改JQM页面数据时, 通常需要再进行一次refresh。可以根据不同的类型选择以下对应的方法。
$(‘div#id‘).trigger(‘refresh‘); $(‘ul#id‘).listview(‘refresh‘); $(‘button#id‘).button(‘refresh‘); $(‘input#id[type=checkbox]‘).checkboxradio(‘refresh‘);
3. tap
JQueryMobile在Android设备上的tap事件会出现多次触发的问题, 对此可以参考Ghost Click。
我们的解决方案是使用Google FastButton,
将原来需要用tap的地方改用fastbutton处理。
4. taphold
检查jquery.mobile-1.2.0.js, 1722行。
timer = setTimeout( function() { triggerCustomEvent( thisObject, "taphold", $.Event( "taphold", { target: origTarget } ) ); }, $.event.special.tap.tapholdThreshold );
在触发taphold事件时没有清除handlers, 所以当taphold事件后, 本不应该被触发的tap事件也被触发了。
暂时修复的方案是在1722行添加:
timer = setTimeout( function() { clearTapHandlers(); // <---- + 这一行 triggerCustomEvent( thisObject, "taphold", $.Event( "taphold", { target: origTarget } ) ); }, $.event.special.tap.tapholdThreshold );
这个bug存在于JQM1.2.0及以下版本。
5. swipe
JQM的swipe响应也是非常慢/诡异, 如果需要使用swipe事件, 建议寻找其他代替的方案, 如TouchSwipe。
6. popup
你可以选择在脚本中生成popup, 并通过popup(‘open‘)
及popup(‘close‘)
进行打开/关闭, 借此可以实现很多实用的功能。
如以下模拟confirm的效果:
var confirm = function (content, title, response) { var html = "
" + title + "
" + content + "取消确定
", previous = $(".ui-popup-active div[data-role=popup]"), divConfirm = $("div#mToast_confirm"); previous.popup(‘close‘); if (divConfirm.length > 0) { divConfirm.remove(); } divConfirm = $(html).appendTo("div[data-role=page]:first"); divConfirm.trigger(‘create‘) // <-- 生成popup .trigger(‘refresh‘) .popup() .find("#mToast_confirm_response").on(‘fastClick‘, function () { divConfirm.popup(‘close‘); previous.popup(‘open‘); response(); }); divConfirm.popup(‘open‘); // --> }; confirm(‘are you sure?‘, ‘Confirm‘, function () { alert(‘sure‘); });
需要注意的是popup(‘open‘)
前需要对div进行.trigger(‘create‘).trigger(‘refresh‘).popup()
。
此外, $.mobile.popup.active
也非常实用, $.mobile.popup.active.element[0]
将返回当前显示的popup层对象。
7. data-rel=back
当你发现使用data-rel=back
的返回按钮响应速度难以忍受的时候, 可以为这个按钮单独绑定一个事件处理器。
如以下脚本将加快header上的返回按钮响应速度:
$(document).bind(‘pageinit‘, function(){ $("div[data-role=page] > div[data-role=header] > a[data-rel=back]").bind( "fastClick", function( event ) { $.mobile.back(); return false; }); });
8. BackButton (PhoneGap + JQM)
在PhoneGap+JQM的方案下, 发现Android手机上的返回硬键无效或者对popup的处理不正确时(多为ajaxEnable=false的情况), 加入以下脚本:
document.addEventListener("deviceready", backKeyListener, false); function backKeyListener() { document.addEventListener("backbutton", onBackKeyDown, false); function onBackKeyDown() { try { if ($.mobile.popup.active) { var popupDiv = $.mobile.popup.active.element; popupDiv.each(function () { if ($(this).parent().hasClass(‘ui-popup-active‘)) { $(this).popup(‘close‘); return false; } }); } else { $.mobile.back(); return false; } } catch (e) { console.log(‘BackButton Listener Catch : ‘ + e); } } }
9. $.mobile.loading
建议把$.mobile.showPageLoadingMsg()
以及$.mobile.hidePageLoadingMsg()
的脚本改为$.mobile.loading(‘show‘)
以及$.mobile.loading(‘hide‘)
。
这个方法同样可以配置内容、样式等参数: $.mobile.loading(‘show‘, {text : ‘test‘, theme : ‘a‘});
。
10. $.mobile.back()
如果你是使用PhoneGap + JQueryMobile进行开发, 设定了ajaxEnable=false
, 并且发现$.mobile.back()
无效, 那么请尝试设定phonegapNavigationEnable=true
。
当该值为true时, $mobile.back()会使用nav.app.backHistory();
, 否则使用window.history.back();
。
但这个参数也 仅 建议在ajaxEnable=false的情况下进行设置。
11. ajaxEnable
如果希望使用PhoneGap将应用打包为app, 我的建议是, 尽量使用ajaxEnable=true
, 否则体验十分不好, 而且你还会遇到更多的问题。
12. 页面跳转
请使用$.mobile.changePage()
代替window.location.href
。
如果要刷新当前页面呢? 我的方法是:
$.mobile.changePage($.mobile.activePage.jqmData(‘url‘), {reloadPage : true});
13. 慎重选择JQueryMobile
如你所见, 使用JQM + PhoneGap进行WebApp开发会遇到许多问题。
但JQM目前还是只适合做简单的WebApp, 如果坚持做复杂, 其效率将会十分堪忧。
当然, 如果你选择了一个正确的方式, 那其中大部分都可以避免。
此外, Github上有许多项目对基于JQM的开发会有很大的帮助。
The-M-Project的UI基于JQM, 但其拥有更好的结构, 并实现了一些你可能会需要的功能。其文档也十分健全, 你可以查看其更详细的介绍。你不一定使用这个框架, 但在JQM的开发上, 这个项目有许多值得借鉴的地方。
离线数据的库, 这里有一个结合JQM的Demo。
3. artTemplate