手機(jī)互動(dòng)網(wǎng)頁項(xiàng)目設(shè)計(jì)總結(jié)(下)
一、項(xiàng)目地址
1、TGA移動(dòng)游戲官網(wǎng)
TGA城市拉力賽
2、UP+邀請函
3、天天酷跑里約進(jìn)擊版
二、布局
1、按屏滾動(dòng)會(huì)碰到什么問題?
問題:
按屏滾動(dòng)即用戶滑動(dòng)屏幕翻頁,屏幕自動(dòng)鎖定到當(dāng)前頁。我們會(huì)碰到的問題:
1)、頁面100%高度包含地址欄高度,當(dāng)?shù)刂窓诖嬖跁r(shí),會(huì)有部分內(nèi)容被隱藏,如下圖所示:
2)、如何阻止瀏覽器默認(rèn)滾動(dòng)事件更好?
解決:
1)、重置<html>高度;
document.documentElement.style.height = window.innerHeight + 'px';
2)、默認(rèn)事件禁止touchmove比touchstart更好,否則還需要單獨(dú)處理<a><input>等問題。
document.documentElement.addEventListener('touchmove', function(e){ e.preventDefault(); });
2、flexbox會(huì)碰到什么問題?
問題:
自適應(yīng)高度/寬度的flex元素內(nèi),子元素尺寸以百分比為單位(如height:100%)無效。
解決:
需要將子元素設(shè)為絕對(duì)定位。
3、fixed定位需要慎用。
問題:
在ios系統(tǒng)的瀏覽器中,頁面加載時(shí)偶爾會(huì)出現(xiàn)抖動(dòng)情況。
解決:
使用絕對(duì)定位或flexbox模擬解決。
三、動(dòng)畫
1、CSS動(dòng)畫與Canvas動(dòng)畫性能優(yōu)劣分析
問題:
目前移動(dòng)端動(dòng)畫,特別在H5游戲與強(qiáng)交互性網(wǎng)站應(yīng)用非常普遍,常用的實(shí)現(xiàn)方式是CSS(硬件加速)與Canvas,但缺乏完整的性能分析數(shù)據(jù),我們在選擇動(dòng)畫實(shí)現(xiàn)方式時(shí)還是存在糾結(jié)。
解決:
選用高中低不同配置不同系統(tǒng)的4款設(shè)備,進(jìn)行了一輪測試:
1)、測試DEMO:
Canvas: 2個(gè)元素 | Canvas:10個(gè)元素 |
DOM:2個(gè)元素 | DOM:10個(gè)元素 |
2)、測試數(shù)據(jù):
紅色代表警告數(shù)據(jù),一般60以上較為順暢。
特征 測試機(jī)型 |
iPhone 4s (iOS6) |
iTouch 4 (iOS5) |
三星i9250 (Android 4.0) |
三星gt s6358 (Android 2.2) |
預(yù)計(jì)市場份額 | 中高端(>50%) | 中端(35%) | 低端(<15%) | |
Canvas: 2個(gè)元素 | 110 | 105 | 95(丟幀) | 20 |
Canvas:10個(gè)元素 | 110 | 65 | 65(丟幀) | 10 |
DOM:2個(gè)元素 | 115 | 80 | 100 | 12 |
DOM:10個(gè)元素 | 60 | 30 | 50 | 5 |
3)、測試結(jié)論:
a、CSS動(dòng)畫更為流暢、但內(nèi)存占用過高,動(dòng)畫元素在5個(gè)以內(nèi)更為推薦;
b、Canvas動(dòng)畫存在丟幀現(xiàn)象,這一現(xiàn)象在android中低端手機(jī)中表現(xiàn)更為明顯;
c、5個(gè)以內(nèi)動(dòng)畫元素,選用CSS動(dòng)畫,80%的設(shè)備幀頻可達(dá)80以上。
2、當(dāng)縮放動(dòng)畫碰上硬件加速
問題:
如下圖所示,如何平滑地實(shí)現(xiàn)底部導(dǎo)航的縮放呢?
1)、不能直接用scale來縮放,這樣會(huì)造成文字的縮放與虛化;
2)、不能通過 height的變化來更改高度,這樣不能啟用硬件加速。
解答:
需要分拆實(shí)現(xiàn),高度變化用translateY來做,單獨(dú)對(duì)里面的圖標(biāo)使用scale。
代碼:
nav{transition:.5s all ;} nav .icon{transition:.5s all;transform-origin:center bottom;} nav.mini{transform: translate(0, 50px);} nav.mini .icon{transform: scale(0.5, 0.5);}
3、動(dòng)畫由Javascript來維護(hù)還是css維護(hù)的思考
近期制作的幾個(gè)動(dòng)畫較多的網(wǎng)站,在動(dòng)畫的調(diào)整與維護(hù)上花了不少時(shí)間,于是有了以下思考。
1)、判斷優(yōu)劣維度
開發(fā)效率、維護(hù)、性能
2)、具體分析
a、由CSS維護(hù)。
實(shí)現(xiàn):
結(jié)合transition和animation,將動(dòng)畫時(shí)間軸寫死在CSS中。
開發(fā)效率:高。
結(jié)合現(xiàn)有配套工具,能夠快速完成動(dòng)畫制作。
維護(hù):差。
如果動(dòng)畫涉及回調(diào)函數(shù)執(zhí)行,時(shí)間軸同時(shí)存在于CSS與Javascript中,維護(hù)成本會(huì)大大提升。如果涉及判斷與循環(huán),動(dòng)畫效果將同時(shí)由CSS與Javascript來維護(hù),代價(jià)太高。
性能:高。
動(dòng)畫無需解析Javascript,減少DOM操作,性能較Javascript的方式更高。
b、由Javascript維護(hù)。
封裝動(dòng)畫框架,底層調(diào)用CSS動(dòng)畫,時(shí)間軸由Javascript來維護(hù)。
開發(fā)效率:高。無需考慮各平臺(tái)兼容性,配置參數(shù)即可完成動(dòng)畫,對(duì)于復(fù)雜動(dòng)畫,也可開發(fā)配套工具。
維護(hù):好。動(dòng)畫效果、動(dòng)畫邏輯,全部由Javascript來維護(hù),動(dòng)畫可控性更高,維護(hù)變得很輕松。
性能:較高。由于底層還是調(diào)用CSS動(dòng)畫,DOM操作次數(shù)有限,性能損耗并不會(huì)太高。
3)、總結(jié)。
對(duì)于弱交互,無邏輯的動(dòng)畫,可以選用CSS的方式。對(duì)于復(fù)雜動(dòng)畫,更推薦由Javascript來維護(hù)。
四、特性
1、如何解決音頻播放預(yù)加載與延時(shí)的問題
問題:
通過對(duì)各個(gè)平臺(tái)下進(jìn)行音頻特性測試,存在問題如下:
IOS6 | Android2.3 | Android4.0 | |
Audio標(biāo)簽base64 | 不支持 | 不支持 | 不支持 |
預(yù)加載 | 不支持 | 支持 | 支持 |
自動(dòng)播放 | 不支持 | 支持 | 支持 |
不同聲音同時(shí)播放 | 支持 | 支持 | 支持 |
播放延時(shí) | 無 | 無 | 無 |
Web audio | 支持 | 不支持 | 不支持 |
解決:
問題/平臺(tái) | IOS4-5 | IOS6-7 | Android2.3-4.2 |
預(yù)加載 | 不支持 |
Web audio (Base64-> ArrayBuffer) |
Audio對(duì)象 |
代碼:
封裝類:
var lib = { _keyStr : "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=", decodeArrayBuffer: function(input) { var bytes = (input.length/4) * 3; var ab = new ArrayBuffer(bytes); this.decode(input, ab); return ab; }, decode: function(input, arrayBuffer) { var lkey1 = this._keyStr.indexOf(input.charAt(input.length-1)); var lkey2 = this._keyStr.indexOf(input.charAt(input.length-2)); var bytes = (input.length/4) * 3; if (lkey1 == 64) bytes--; if (lkey2 == 64) bytes--; var uarray; var chr1, chr2, chr3; var enc1, enc2, enc3, enc4; var i = 0; var j = 0; if (arrayBuffer) uarray = new Uint8Array(arrayBuffer); else uarray = new Uint8Array(bytes); input = input.replace(/[^A-Za-z0-9+/=]/g, ""); for (i=0; i<bytes; i+=3) { enc1 = this._keyStr.indexOf(input.charAt(j++)); enc2 = this._keyStr.indexOf(input.charAt(j++)); enc3 = this._keyStr.indexOf(input.charAt(j++)); enc4 = this._keyStr.indexOf(input.charAt(j++)); chr1 = (enc1 << 2) | (enc2 >> 4); chr2 = ((enc2 & 15) << 4) | (enc3 >> 2); chr3 = ((enc3 & 3) << 6) | enc4; uarray[i] = chr1; if (enc3 != 64) uarray[i+1] = chr2; if (enc4 != 64) uarray[i+2] = chr3; } return uarray; }, getScript: function(url, callback, charset){ var head = document.getElementsByTagName("head")[0] || document.documentElement, script = document.createElement("script"); script.src = url; charset && (script.charset = charset); script.onload = script.onreadystatechange = function() { if ((!this.readyState || this.readyState === "loaded" || this.readyState === "complete") ) { callback && callback(); script.onload = script.onreadystatechange = null; if (head && script.parentNode) head.removeChild(script); } }; head.appendChild(script); return script; } } var QAudio = function(config){ this.config = config; this.init(); } QAudio.prototype.init = function(){ var config = this.config; this.src = config.src; this.base64 = config.base64; if ('AudioContext' in window) { this.audioContext = new AudioContext(); } else if ('webkitAudioContext' in window) { this.audioContext = new webkitAudioContext(); } else { this.audio = document.createElement('audio'); this.audio.preload = 'auto'; this.audio.src = this.src; } // base64數(shù)據(jù)處理 if(this.audioContext) { this._handleBase64(); } this.load(); } QAudio.prototype._handleBase64 = function(){ var self = this; window[this.config.callback] = function(data){ var arrayBuff = lib.decodeArrayBuffer(data); self.audioContext.decodeAudioData(arrayBuff, function(audioData) { self._arrayBuff = audioData; }); } } QAudio.prototype.load = function(){ // 如果支持Web Audio,加載base64音頻 if(this.audioContext) { lib.getScript(this.base64); } else if(this.audio) { this.audio.load(); } } QAudio.prototype.play = function(){ if(this.audioContext) { var mySource = this.audioContext.createBufferSource(); mySource.buffer = this._arrayBuff; mySource.connect(this.audioContext.destination); if ('AudioContext' in window) { mySource.start(0); } else if ('webkitAudioContext' in window) { mySource.noteOn(0); } } else if(this.audio) { this.audio.play(); } }
調(diào)用:
new QAudio({src: './audio/pop1.mp3', base64: './audio/pop1.js', callback: 'cbPop1'});
示例:
2、重力感應(yīng)
問題:
重力感應(yīng)的實(shí)現(xiàn)相對(duì)簡單,具體見如下代碼。值得注意的是,android需3.0及以上才支持。
解決:
代碼:
var box = document.querySelector('.box'); var timer; window.addEventListener('deviceorientation', function(e) { var x = parseInt(e.gamma); var y = parseInt(e.beta); var r = parseInt(e.alpha); box.style.webkitTransform = 'translate3d(' + x*window.innerWidth/180 + 'px, '+ y*window.innerHeight/180 +'px, 0) rotate('+ r +'deg)' });
3、啟動(dòng)原生應(yīng)用
背景:
在全民飛機(jī)大戰(zhàn)移動(dòng)版的開發(fā)中,需要滿足一個(gè)需求,當(dāng)用戶點(diǎn)擊下載按鈕時(shí),如果用戶已經(jīng)安裝了該APP,則直接啟動(dòng),否則跳到下載頁。
問題:
在網(wǎng)頁中如何啟動(dòng)原生應(yīng)用?
解決:
首先通過URL Scheme嘗試打開本地APP,如果打開成功,網(wǎng)頁中setTimeout將會(huì)掛起,重新回到頁面時(shí)超時(shí)不再執(zhí)行,否則,如果打開失敗,則嘗試下載或進(jìn)入下載頁面。
代碼:
(function(){ var iosURL = 'itms-apps://itunes.apple.com/cn/app/quan-min-fei-ji-da-zhan/id731871690?l=en&mt=8'; var androidURL = 'http://dlied5.qq.com/wefly/release/android/1.0.5/wefly_1.0.5.6_android_8CF5F95E.apk'; var iosScheme = 'tencentlaunch100539858:'; var androidScheme = 'webwx47a71dd8a1875943:'; var homePage = '/web201404/'; if(/iphone|ipod|ipad/i.test(navigator.userAgent)) { window.location = iosScheme; var startDate = new Date(); setTimeout(function(){ if(new Date() - startDate > 1000) return; window.location= iosURL; window.setTimeout(function(){ window.location= homePage; }, 100) } , 800); } else if(/android/i.test(navigator.userAgent)) { window.location = androidScheme; var startDate = new Date(); setTimeout(function(){ if(new Date() - startDate > 1000) return; window.location= androidURL; window.setTimeout(function(){ window.location= homePage; }, 1000) } , 800); } else { alert('非常抱歉,本游戲暫時(shí)只提供android和ios版。'); window.location= homePage; } })();
4、地理定位 HTML 5 Geolocatioin
地理定位是HTML 5的重要熱性之一,它提供了用戶的確切位置(經(jīng)緯度)。借助這個(gè)特性我們能夠開發(fā)基于地理位置的頁面。
HTML 5 Geolocatioin API 使用起來很簡單,我們通過下面的方式調(diào)用。
ar x=document.getElementById("demo"); function getLocation(){ if (navigator.geolocation){ navigator.geolocation.getCurrentPosition(showPosition); }else{x.innerHTML="Geolocation is not supported by this browser.";} } function showPosition(position){ x.innerHTML="Latitude: " + position.coords.latitude + "<br />Longitude: " + position.coords.longitude; }
代碼邏輯是先判斷你的設(shè)備支不支持這個(gè)接口,進(jìn)而給出相應(yīng)的經(jīng)緯度。結(jié)合TGA城市賽這個(gè)案例,我們可以通過用戶在訪問頁面的時(shí)候,獲取其經(jīng)緯度,判斷用戶所在的城市,自動(dòng)為其選中地址,減少操作步驟。如圖:
地址:
這里有一個(gè)小技巧,使用這個(gè)API我們只能獲取到經(jīng)緯度,那怎么轉(zhuǎn)換成對(duì)應(yīng)的具體地址呢?
1. 可以使用google的地圖API:
http://maps.google.com/maps/api/geocode/json?latlng=39.9,41.033&language=zh-CN&sensor=true
只要傳入正確的經(jīng)緯度,就能夠返回相應(yīng)的具體到街道的json數(shù)據(jù)。
2. 通過我們騰訊地圖的地址解析,可以做到具體地理位置字符跟經(jīng)緯度的轉(zhuǎn)換。
http://open.map.qq.com/javascript_v2/case-run.html#sample-geocoding-simple。
五、其它
1、字體圖標(biāo)應(yīng)用設(shè)計(jì)環(huán)節(jié)的打通
問題:
字體圖標(biāo)在這個(gè)項(xiàng)目中大量應(yīng)用,但面臨的一個(gè)問題是,輔助前端開發(fā)的配套工具已經(jīng)非常成熟,但設(shè)計(jì)環(huán)節(jié)的打通一直是一片空白。設(shè)計(jì)師在應(yīng)用字體圖標(biāo)時(shí)還是只能以最原始的方式將圖標(biāo)拖入設(shè)計(jì)稿,十分不便。
解決:
為設(shè)計(jì)師提供一款Photoshop插件,將字體文件導(dǎo)入插件,設(shè)計(jì)師可以直接點(diǎn)擊圖標(biāo)插入設(shè)計(jì)稿,將字體圖標(biāo)應(yīng)用的設(shè)計(jì)環(huán)節(jié)打通,該插件正在開發(fā)中。
2、當(dāng)transform碰上模糊
問題:
如下圖ball1所示,在android中,如果元素或其父元素應(yīng)用transform后,元素設(shè)置border-radius會(huì)變模糊。
解決:
將元素放大一倍后,再進(jìn)行縮放(如果該元素后有文本節(jié)點(diǎn),也可避免模糊)
body{padding: 20px;background:purple;-webkit-transform: translate3d(0px, 0px, 0px);} .ball1{width: 50px;height: 50px;border-radius:25px;margin-top: 20px;background: #fff;} .ball2{width: 100px;height: 100px;border-radius:50px;margin-top: 20px;background: #fff;-webkit-transform-origin:0 0;-webkit-transform: scale(0.5, 0.5);-webkit-backface-visibility:hidden;}
DEMO地址:
http://sy.qq.com/brucewan/feature/blur-radius.html
六、數(shù)據(jù)
1、用戶環(huán)境
從TGA移動(dòng)游戲官網(wǎng)統(tǒng)計(jì)數(shù)據(jù)分析:
1)、抽樣50%訪問量統(tǒng)計(jì),移動(dòng)端用戶平均加速速度為70Kb/s;
2)、總加載時(shí)間為4.12s,86%的用戶愿意等待頁面加載完畢,52%的用戶抵達(dá)第二屏;
3)、這次滾屏支持用戶點(diǎn)擊游戲圖標(biāo)或手指滑動(dòng),但只有7%的人會(huì)使用點(diǎn)擊操作,我們后續(xù)項(xiàng)目的設(shè)計(jì)中可以得到啟發(fā)。
本文地址:http://pkvc.cn/tutorial/id1998.html
您可能還喜歡
- 關(guān)于第三方注冊和本地注冊的選擇
- 8大網(wǎng)頁前端界面必備jQuery插件
- 掌上指路標(biāo) —– APP架構(gòu)與導(dǎo)航設(shè)計(jì)
- 時(shí)尚電商網(wǎng)站交互分析
- 軟件推薦為移動(dòng)設(shè)計(jì)而生-Justinmind
- TGideas實(shí)例:加載,不只是少一點(diǎn)點(diǎn)
- 網(wǎng)頁設(shè)計(jì)中引人入勝的引導(dǎo)頁設(shè)計(jì)
- 體驗(yàn)新版Apple AppStore ——App設(shè)計(jì)從
- 確認(rèn)按鈕在左邊,取消按鈕在右邊?
- 交互設(shè)計(jì)從登錄開始
- 專訪:石墨文檔產(chǎn)品總監(jiān)羅穎
- UI設(shè)計(jì)不得不知的移動(dòng)端UI尺寸適
- 光音移動(dòng)設(shè)計(jì)規(guī)范 — 表單類
- 體驗(yàn)設(shè)計(jì)中的排序問題
- 網(wǎng)頁設(shè)計(jì)精粹 網(wǎng)頁中那些迷人的按
- aliued:響應(yīng)式設(shè)計(jì)的現(xiàn)狀與趨勢
- 10個(gè)智能對(duì)象處理的ps技巧
- 網(wǎng)頁UI - 原子設(shè)計(jì)理論(上)
- 如何通過設(shè)計(jì)提升banner點(diǎn)擊率?
- 晉小彥視覺設(shè)計(jì)系列文章(二):全屏