精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

微信小程序架構(gòu)分析 (下)

開發(fā) 架構(gòu)
這一篇拖了一段時(shí)間,原因是實(shí)現(xiàn)一個(gè)可以運(yùn)行微信小程序的 web 環(huán)境比我想象中要困難一些, 這一方面是因?yàn)槲⑿艑τ诖a進(jìn)行了壓縮混淆,另一方面主要原因是開發(fā)者工具內(nèi)部邏輯調(diào)用比較復(fù)雜(難怪 bug 不少),完全無法拿出來重用。

[[193510]]

【引自第九程序的博客】這一篇拖了一段時(shí)間,原因是實(shí)現(xiàn)一個(gè)可以運(yùn)行微信小程序的 web 環(huán)境比我想象中要困難一些, 這一方面是因?yàn)槲⑿艑τ诖a進(jìn)行了壓縮混淆,另一方面主要原因是開發(fā)者工具內(nèi)部邏輯調(diào)用比較復(fù)雜(難怪 bug 不少),完全無法拿出來重用。

小程序?qū)崟r(shí)運(yùn)行工具 wept 的開發(fā)已經(jīng)基本完成了, 你可以通過我的代碼對小程序的 web 環(huán)境實(shí)現(xiàn)有更全面的認(rèn)識。下面我將介紹它的實(shí)現(xiàn)過程以及實(shí)時(shí)更新的原理。

小程序 web 服務(wù)實(shí)現(xiàn)

我在 wept 的開發(fā)中使用 koa 提供 web 服務(wù),以及 et-improve 提供模板渲染。

***步: 準(zhǔn)備頁面模板

我們需要三個(gè)頁面,一個(gè)做為控制層 index.html,一個(gè)做為 service 層service.html,還有一個(gè)做為 view 層的 view.html

index.html:

  1. <div class="head"
  2. </div> 
  3. <div class="scrollable"
  4. </div> 
  5. <div class="tabbar-root"
  6. </div> 
  7. <script> 
  8.   var __wxConfig__ = {{= _.config}} 
  9.   var __root__ = '{{= _.root}}' 
  10. </script> 
  11. <script src="/script/build.js"></script>  

service.html:

  1. <head> 
  2.   <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 
  3.   <link href="https://res.wx.qq.com/mpres/htmledition/images/favicon218877.ico" rel="Shortcut Icon"
  4.   <script> 
  5.   var __wxAppData = {} 
  6.   var __wxRoute 
  7.   var __wxRouteBegin 
  8.   global = {} 
  9.   var __wxConfig = {{= _.config}} 
  10.   </script> 
  11.   <script src="/script/bridge.js" type="text/javascript"></script> 
  12.   <script src="/script/service.js" type="text/javascript"></script> 
  13.   {{each _.utils as util}} 
  14.   <script src="/app/{{= util}}" type="text/javascript"></script> 
  15.   {{/}} 
  16.   <script src="/app/app.js" type="text/javascript"></script> 
  17.   {{each _.routes as route}} 
  18.   <script> var __wxRoute = '{{= route | noext}}', __wxRouteBegin = true;</script> 
  19.   <script src="/app/{{= route}}" type="text/javascript"></script> 
  20.   {{/}} 
  21. </head> 
  22. <body> 
  23.   <script> 
  24.     window._____sendMsgToNW({ 
  25.       sdkName: 'APP_SERVICE_COMPLETE' 
  26.     }) 
  27.   </script> 
  28. </body> 

view.html:

  1. <head> 
  2.   <link href="https://res.wx.qq.com/mpres/htmledition/images/favicon218877.ico" rel="Shortcut Icon"
  3.   <meta charset="UTF-8" /> 
  4.   <meta name="viewport" content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0" /> 
  5.   <link rel="stylesheet" type="text/css" href="/css/default.css"
  6.   <link rel="stylesheet" type="text/css" href="/app/app.wxss"
  7.   <link rel="stylesheet" type="text/css" href="/app/{{= _.path}}.wxss"
  8.   <script> var __path__ = '{{= _.path}}'</script> 
  9.   <script src="/script/ViewBridge.js" async type="text/javascript"></script> 
  10.   <script src="/script/view.js" type="text/javascript"></script> 
  11.   <script> 
  12.   {{= _.inject_js}} 
  13.   </script> 
  14.   <script> 
  15.     document.dispatchEvent(new CustomEvent("generateFuncReady", { 
  16.       detail: { 
  17.         generateFunc: $gwx('./{{= _.path}}.wxml'
  18.       } 
  19.     })) 
  20.   </script> 
  21. </head> 
  22. <body> 
  23.   <div></div> 
  24. </body>  

第二步: 實(shí)現(xiàn) http 服務(wù)

用 koa 實(shí)現(xiàn)的代碼邏輯非常簡單:

server.js

  1. // 日志中間件 
  2. app.use(logger()) 
  3. // gzip 
  4. app.use(compress({ 
  5.   threshold: 2048, 
  6.   flush: require('zlib').Z_SYNC_FLUSH 
  7. })) 
  8. // 錯(cuò)誤提醒中間件 
  9. app.use(notifyError) 
  10. // 使用當(dāng)前目錄下文件處理 404 請求 
  11. app.use(staticFallback) 
  12. // 各種 route 實(shí)現(xiàn) 
  13. app.use(router.routes()) 
  14. app.use(router.allowedMethods()) 
  15. // 對于 public 目錄啟用靜態(tài)文件服務(wù) 
  16. app.use(require('koa-static')(path.resolve(__dirname, '../public'))) 
  17. // 創(chuàng)建啟動(dòng)服務(wù) 
  18. let server = http.createServer(app.callback()) 
  19. server.listen(3000)  

router.js

  1. router.get('/'function *() { 
  2.   // 加載 index.html 模板和數(shù)據(jù),輸出 index 頁面 
  3. }) 
  4.  
  5. router.get('/appservice'function *() { 
  6.   // 加載 service.html 模板和數(shù)據(jù),輸出 service 頁面 
  7. }) 
  8.  
  9. // 讓 `/app/**` 加載小程序所在目錄文件 
  10. router.get('/app/(.*)'function* () { 
  11.   if (/\.(wxss|js)$/.test(file)) { 
  12.     // 動(dòng)態(tài)編譯為 css 和相應(yīng) js 
  13.   } else if (/\.wxml/.test(file)) { 
  14.     // 動(dòng)態(tài)編譯為 html 
  15.   } else { 
  16.     // 查找其它類型文件, 存在則返回 
  17.     let exists = util.exists(file) 
  18.     if (exists) { 
  19.       yield send(this, file) 
  20.     } else { 
  21.       this.status = 404 
  22.       throw new Error(`File: ${file} not found`) 
  23.     } 
  24.   } 
  25. })  

第三步:實(shí)現(xiàn)控制層功能

實(shí)現(xiàn)完上面兩步,就可以訪問 view 頁面了,但是你會(huì)發(fā)現(xiàn)它只能渲染,并不會(huì)有任何功能,因?yàn)?view 層功能依賴于控制層進(jìn)行的通訊, 如果控制層收不到消息,它不會(huì)響應(yīng)任何事件。

控制層是整個(gè)實(shí)現(xiàn)過程中最復(fù)雜的一塊,因?yàn)楣俜焦ぞ叩拇a與 nwjs 以及 react 等第三方組件耦合過高,所以無法拿來直接使用。 你可以在 wept 項(xiàng)目的 src 目錄下找到控制層邏輯的所有代碼,總體上控制層要負(fù)責(zé)以下幾個(gè)功能:

  • 實(shí)現(xiàn) service 層,view 層以及控制層之間的通訊邏輯
  • 依據(jù)路由指令動(dòng)態(tài)創(chuàng)建 view (wept 使用 iframe 實(shí)現(xiàn))
  • 根據(jù)當(dāng)前頁面動(dòng)態(tài)渲染 header 和 tabbar
  • 實(shí)現(xiàn)原生 API 調(diào)用,返回結(jié)果給 service 層

wept 里面 iframe 之間的通訊是通過 message.js 模塊實(shí)現(xiàn)的,控制頁面(index.html)代碼如下:

  1. window.addEventListener('message'function (e) { 
  2.   let data = e.data 
  3.   let cmd = data.command 
  4.   let msg = data.msg 
  5.   // 沒有跟 contentscript 握手階段,不需要處理 
  6.   if (data.to == 'contentscript'return 
  7.   // 這是個(gè)遺留方法,基本廢棄掉了 
  8.   if (data.command == 'EXEC_JSSDK') { 
  9.     sdk(data) 
  10.   // 直接轉(zhuǎn)發(fā) view 層消息到 service,主要是各種事件通知 
  11.   } else if (cmd == 'TO_APP_SERVICE') { 
  12.     toAppService(data) 
  13.   // 除了 publish 發(fā)送消息給 view 層以及控制層可以處理的邏輯(例如設(shè)置標(biāo)題), 
  14.   // 其它全部轉(zhuǎn)發(fā) service 處理,所有控制層的處理結(jié)果統(tǒng)一先返回 service 
  15.   } else if (cmd == 'COMMAND_FROM_ASJS') { 
  16.     let sdkName = data.sdkName 
  17.     if (command.hasOwnProperty(sdkName)) { 
  18.       command[sdkName](data) 
  19.     } else { 
  20.       console.warn(`Method ${sdkName} not implemented for command!`) 
  21.     } 
  22.   } else { 
  23.     console.warn(`Command ${cmd} not recognized!`) 
  24.   } 
  25. })  

具體實(shí)現(xiàn)邏輯可以查看 src/command.js src/service.jssrc/sdk/*.js。對于 view/service 頁面只需把原來 bridge.js 的window.postMessage 改為 window.top.postMessage 即可。

view 層的控制邏輯由 src/view.js 以及 src/viewManage.js 實(shí)現(xiàn),viewManage 實(shí)現(xiàn)了 navigateTo, redirectTo 以及 navigateBack 來響應(yīng) service 層通過名為 publish 的 command 傳來的對應(yīng)頁面路由事件。

header.js 和 tabbar.js 包含了基于 react 實(shí)現(xiàn)的 header 和 tabbar 模塊(原計(jì)劃是使用 vue,但是沒找到與原生 js 模塊通訊的 API)

sdk 目錄下包含了 storage,錄音,羅盤模塊,其它比較簡單一些的原生底層調(diào)用我直接寫在 command.js 里面了。

以上就是實(shí)現(xiàn)運(yùn)行小程序所需 webserver 的全部邏輯了,其實(shí)現(xiàn)并不復(fù)雜,主要困難在與理解微信這一整套通訊方式。

實(shí)現(xiàn)小程序?qū)崟r(shí)更新

***步: 監(jiān)視文件變化并通知前端

wept 使用了 chokidar 模塊監(jiān)視文件變化,變化后使用 WebSocket 告知所有客戶端進(jìn)行更新操作。 具體實(shí)現(xiàn)位于 lib/watcher.js 和 lib/socket.js, 發(fā)送內(nèi)容是 json 格式的字符串。

前端控制層收到 WebSocket 消息后再通過 postMessage 接口轉(zhuǎn)發(fā)消息給 view/service 層:

  1. view.postMessage({ 
  2.   msg: { 
  3.     data: { 
  4.       data: { path } 
  5.     }, 
  6.     eventName: 'reload' 
  7.   }, 
  8.   command: 'CUSTOM' 
  9. })  

view/service 層監(jiān)聽 reload 事件:

  1. WeixinJSBridge.subscribe('reload'function(data) { 
  2.   // data 即為上面的 msg.data 
  3. })  

第二步: 前端響應(yīng)不同文件變化

前端需要對 4 種(wxml wxss json javascript)不同類型文件進(jìn)行 4 種不同的熱更新處理,其中 wxss 和 json 相對簡單。

  • wxss 文件變化后前端控制層通知(postMessage 接口)對應(yīng)頁面(如果是 app.wxss 則是所有 view 頁面)進(jìn)行刷新,view 層收到消息后只需要更改對應(yīng) css 文件的時(shí)間戳就可以了,代碼如下:
  1. o.subscribe('reload'function(data) { 
  2.     if (/\.wxss$/.test(data.path)) { 
  3.     var p = '/app/' + data.path 
  4.     var els = document.getElementsByTagName('link'
  5.     ;[].slice.call(els).forEach(function(el) { 
  6.       var href = el.getAttribute('href').replace(/\?(.*)$/, ''
  7.       if (p == href) { 
  8.         console.info('Reload: ' + data.path) 
  9.         el.setAttribute('href', href + '?id=' + Date.now()) 
  10.       } 
  11.     }) 
  12.   } 
  13. })  
  • json 文件變化首先需要判斷,如果是 app.json 我們無法熱更新,所以目前做法是刷新頁面,對于頁面的 json, 我們只需要在控制層上對 header 設(shè)置相應(yīng)狀態(tài)就可以了 (渲染工作由 react 幫我們處理):
  1. socket.onmessage = function (e) { 
  2.   let data = JSON.parse(e.data) 
  3.   let p = data.path 
  4.   if (data.type == 'reload'){ 
  5.     if (p == 'app.json') { 
  6.       redirectToHome() 
  7.     } else if (/\.json$/.test(p)) { 
  8.       let win = window.__wxConfig__['window'
  9.       win.pages[p.replace(/\.json$/, '')] = data.content 
  10.       // header 通過全局 __wxConfig__ 獲取 state 進(jìn)行渲染 
  11.       header.reset() 
  12.       console.info(`Reset header for ${p.replace(/\.json$/, '')}`) 
  13.     } 
  14.   } 
  15.  
  • wxml 使用 VirtualDom API 提供的 diff apply 進(jìn)行處理。首先需要一個(gè)接口獲取新的 generateFunc 函數(shù)(用于生成 VirtualDom), 添加 koa 的 router:
  1. router.get('/generateFunc'function* () { 
  2.   this.body = yield loadFile(this.query.path + '.wxml'
  3.   this.type = 'text' 
  4. }) 
  5.  
  6. function loadFile(p, throwErr = true) { 
  7.   return new Promise((resolve, reject) => { 
  8.     fs.stat(`./${p}`, (err, stats) => { 
  9.       if (err) { 
  10.         if (throwErr) return reject(new Error(`file ${p} not found`)) 
  11.         // 文件不存在有可能是文件被刪除,所以不能使用 reject 
  12.         return resolve(''
  13.       } 
  14.       if (stats && stats.isFile()) { 
  15.         // parer 函數(shù)調(diào)用 exec 命令執(zhí)行 wcsc 文件生成 wxml 對應(yīng)的 javascript 代碼 
  16.         return parser(`${p}`).then(resolve, reject) 
  17.       } else { 
  18.         return resolve(''
  19.       } 
  20.     }) 
  21.   }) 
  22.  
  • 有了接口就可以請求接口,然后執(zhí)行返回函數(shù)進(jìn)行 diff apply:
  1. // curr 為當(dāng)前的 VirtualDom 樹 
  2. if (!curr) return 
  3. var xhr = new XMLHttpRequest() 
  4. xhr.onreadystatechange = function() { 
  5.   if (xhr.readyState === 4) { 
  6.     if (xhr.status === 200) { 
  7.       var text = xhr.responseText 
  8.       var func = new Function(text + '\n return $gwx("./' +__path__+ '.wxml")'
  9.       window.__generateFunc__ = func() 
  10.       var oldTree = curr 
  11.       // 獲取當(dāng)前 data 生成新的樹 
  12.       var o = m(p.default.getData(), false), 
  13.       // 進(jìn)行 diff apply 
  14.       a = oldTree.diff(o); 
  15.       a.apply(x); 
  16.       document.dispatchEvent(new CustomEvent("pageReRender", {})); 
  17.       console.info('Hot apply: ' + __path__ + '.wxml'
  18.     } 
  19.   } 
  20. xhr.open('GET''/generateFunc?path=' + encodeURIComponent(__path__)) 
  21. xhr.send()  
  • javascript 更新邏輯相對復(fù)雜一些, 首先依然是一個(gè)接口來獲取新的 javascript 代碼:
  1. router.get('/generateJavascript'function* () { 
  2.   this.body = yield loadFile(this.query.path) 
  3.   this.type = 'text' 
  4. })  

然后我們在 window 對象上加入 Reload 函數(shù)執(zhí)行具體的更換邏輯:

  1. window.Reload = function (e) { 
  2. var pages = __wxConfig.pages; 
  3. if (pages.indexOf(window.__wxRoute) == -1) return 
  4. // 替換原來的構(gòu)造函數(shù) 
  5. f[window.__wxRoute] = e 
  6. var keys = Object.keys(p) 
  7. // 判定是否當(dāng)前使用中頁面 
  8. var isCurr = s.route == window.__wxRoute 
  9. keys.forEach(function (key) { 
  10.   var o = p[key]; 
  11.   key = Number(key
  12.   var query = o.__query__ 
  13.   var page = o.page 
  14.   var route = o.route 
  15.   // 頁面已經(jīng)被創(chuàng)建 
  16.   if (route == window.__wxRoute) { 
  17.     // 執(zhí)行封裝后的 onHide 和 onUnload 
  18.     isCurr && page.onHide() 
  19.     page.onUnload() 
  20.     // 創(chuàng)建新 page 對象 
  21.     var newPage = new a.default(e, key, route) 
  22.     newPage.__query__ = query 
  23.     // 重新綁定當(dāng)前頁面 
  24.     if (isCurr) s.page = newPage 
  25.     o.page = newPage 
  26.     // 執(zhí)行 onLoad 和 onShow 
  27.     newPage.onLoad() 
  28.     if (isCurr) newPage.onShow() 
  29.     // 更新 data 數(shù)據(jù) 
  30.     window.__wxAppData[route] = newPage.data 
  31.     window.__wxAppData[route].__webviewId__ = key 
  32.     // 發(fā)送更新事件, 通知 view 層 
  33.     u.publish(c.UPDATE_APP_DATA) 
  34.     u.info("Update view with init data"
  35.     u.info(newPage.data) 
  36.     // 發(fā)送 appDataChange 事件 
  37.     u.publish("appDataChange", { 
  38.       data: { 
  39.         data: newPage.data 
  40.       }, 
  41.       option: { 
  42.         timestampDate.now() 
  43.       } 
  44.     }) 
  45.     newPage.__webviewReady__ = true 
  46.   } 
  47. }) 
  48. u.info("Reload page: " + window.__wxRoute) 
  49.  

以上代碼需要添加到 t.pageHolder 函數(shù)后才可運(yùn)行

***在 view 層初始化后把 Page 函數(shù)切換到 Reload 函數(shù)(當(dāng)然你也可以在請求返回 javascript 前把 Page 重命名為 Reload) 。

  1. <body> 
  2. <script> 
  3.   window._____sendMsgToNW({ 
  4.     sdkName: 'APP_SERVICE_COMPLETE' 
  5.   }) 
  6. </script> 
  7. </body>  

總算是把這個(gè)坑填上了。希望通過這一系列的分析帶給前端開發(fā)者更多思路。

責(zé)任編輯:龐桂玉 來源: 第九程序的博客
相關(guān)推薦

2017-06-09 10:40:00

微信小程序架構(gòu)分析

2017-06-09 10:06:54

微信小程序架構(gòu)分析

2021-06-10 10:51:27

程序基礎(chǔ)架構(gòu)

2016-11-04 10:30:17

微信小程序

2017-05-08 15:03:07

微信小程序開發(fā)實(shí)戰(zhàn)

2016-10-20 21:02:12

微信小程序javascript

2017-01-09 10:01:49

微信小程序

2016-09-28 18:10:59

微信程序MINA

2016-11-04 10:31:49

微信程序指南

2016-11-22 11:23:52

微信小程序騰訊微信

2016-09-27 15:40:58

微信程序前端

2016-11-04 10:49:48

微信小程序

2016-09-27 16:38:24

JavaScript微信Web

2016-11-19 18:06:44

微信小程序張小龍

2017-06-27 10:53:32

2016-09-27 20:36:23

微信HttpWeb

2018-07-26 15:16:50

小程序iPhone X甜酸

2017-02-06 13:32:12

微信小程序思想

2016-12-01 17:33:52

微信

2016-11-04 09:55:16

微信小程序
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號

很污很黄的网站| 国产视频一区二区三区在线播放 | 中文字幕第69页| 国产欧美日韩电影| 精品国产成人在线| 亚洲午夜精品久久久久久浪潮| 国产av一区二区三区精品| 亚洲综合国产激情另类一区| 日韩中文字幕在线播放| 男人的天堂影院| 青青久久精品| 五月综合激情网| 尤物一区二区三区| 全部免费毛片在线播放网站| 精品一区二区三区在线观看国产 | 日韩高清精品免费观看| 97精品中文字幕| 日韩电影在线观看中文字幕| 日韩欧美亚洲另类| 伊人色综合一区二区三区影院视频 | 亚洲国产精品无码久久久久高潮| 成人国产网站| 精品国产电影一区| 欧美一区二区视频在线播放| 大片免费播放在线视频| 99久久伊人精品| 亚洲free嫩bbb| 日韩乱码一区二区三区| 久久xxxx精品视频| 97视频在线看| 欧美日韩在线观看成人| 日韩精品久久久久久久电影99爱| 日韩电影第一页| 精品一区二区三区四区五区六区| 日韩免费在线电影| 欧美在线免费播放| 91好吊色国产欧美日韩在线| 欧美xxxbbb| 亚洲精品菠萝久久久久久久| 日本特级黄色大片| 在线观看黄av| 国产欧美视频一区二区三区| 麻豆91蜜桃| 天堂在线一二区| 成人av免费观看| 官网99热精品| 精品久久久久久亚洲综合网站| 久久国产福利国产秒拍| 国产精品视频在线播放| 国产成人麻豆免费观看| 日韩综合一区二区| 国产精品国语对白| 五月激情丁香网| 三级成人在线视频| 日本久久91av| 无码人妻丰满熟妇奶水区码| 丝袜亚洲另类丝袜在线| 国产成人精品在线| 国产天堂第一区| 蜜臀av一区二区在线免费观看 | 一卡二卡三卡在线| 美女脱光内衣内裤视频久久网站| 日韩免费av在线| 真实新婚偷拍xxxxx| 日韩1区2区日韩1区2区| 国产欧美精品日韩| 国产美女免费视频| 国产91丝袜在线观看| 国产成人亚洲欧美| 色吊丝在线永久观看最新版本| 91啪九色porn原创视频在线观看| 裸模一区二区三区免费| h视频网站在线观看| 中文字幕亚洲一区二区av在线| 中国人体摄影一区二区三区| caoporn97在线视频| 亚洲大片精品永久免费| 人妻精品无码一区二区三区 | 亚洲一级影院| 欧美亚洲视频在线观看| 国产在线观看第一页| 国产一区视频导航| 精品国产aⅴ麻豆| 韩国三级av在线免费观看| 国产精品国产三级国产aⅴ原创| 免费观看黄色的网站| 好久没做在线观看| 色狠狠色噜噜噜综合网| 91香蕉视频免费看| 日韩精品免费一区二区三区竹菊 | 免费一级做a爰片久久毛片潮| 清纯唯美亚洲综合一区| 欧美日韩国产成人在线| 亚洲欧美另类在线视频| 国产在线视视频有精品| 久久久一本精品99久久精品66| 91精品国产91久久久久游泳池| 亚洲精品美国一| 欧美激情国产精品日韩| 亚洲精品a区| 伊人男人综合视频网| 国产精品 欧美激情| 免费在线成人| 147欧美人体大胆444| 黄色小视频在线免费观看| 亚洲精品亚洲人成人网在线播放| 农村妇女精品一二区| 视频一区在线| 中文字幕不卡在线视频极品| 国产一级视频在线观看| 美腿丝袜亚洲色图| 鲁鲁视频www一区二区| 国产三级在线播放| 在线看日本不卡| 800av在线播放| 欧美精品色网| 91丝袜美腿美女视频网站| 国产一级二级三级在线观看| 亚洲国产精品一区二区www在线 | 亚洲日日夜夜| 亚洲欧美日韩爽爽影院| 国产午夜福利精品| 国产馆精品极品| 在线日韩av永久免费观看| 中文字幕在线官网| 日韩av一区在线观看| 欧美精品一区二区蜜桃| 国产一区二区三区精品欧美日韩一区二区三区 | 国产精品视频在线观看| 欧美亚洲在线视频| 国产视频不卡在线| 激情婷婷欧美| 91黄色国产视频| 日本在线免费| 欧美一a一片一级一片| 无码人妻精品一区二区三区温州| 激情亚洲网站| y111111国产精品久久婷婷| 91精彩视频在线观看| 色婷婷亚洲综合| 国产精品亚洲无码| 国产日韩1区| 国产美女精品久久久| 污网站在线免费看| 日韩一区二区三区观看| 夫妻性生活毛片| 久久99九九99精品| 亚洲人成网站在线观看播放| 欧美与亚洲与日本直播| 一区二区三欧美| 奴色虐av一区二区三区| 国产亚洲女人久久久久毛片| 成人免费xxxxx在线视频| 精品欧美久久| 国产日韩欧美中文| 国产一二区在线| 欧美一区二区久久| 久久久久久久蜜桃| a在线播放不卡| 欧美色图另类小说| 国产videos久久| 国产精品女人久久久久久| 男人的天堂在线视频免费观看 | 国产欧美日韩在线一区二区| 国产成+人+综合+亚洲欧洲| 欧美日韩国产综合视频| 欧美午夜精品电影| 永久免费看片直接| 国产不卡视频在线播放| 欧美亚洲一二三区| 国产一区99| 亚洲精品免费网站| 99久久精品免费看国产小宝寻花| 日韩成人黄色av| 97人妻精品视频一区| 亚洲欧洲韩国日本视频| 亚洲精品激情视频| 久久av在线| 日本精品免费视频| 久久久精品国产**网站| 国产精品日韩在线观看| 在线观看中文字幕的网站| 亚洲国产精品成人va在线观看| 天天操夜夜操视频| 亚洲天天做日日做天天谢日日欢| 国产精品99精品无码视亚| 国产欧美一区二区三区国产幕精品| 天堂精品一区二区三区| 日韩中文字幕在线一区| 欧美一区三区三区高中清蜜桃| youjizz在线播放| 日韩精品一区二区三区老鸭窝| 欧美精品二区三区| 最近中文字幕一区二区三区| 国产精品久久AV无码| 看片的网站亚洲| 鲁一鲁一鲁一鲁一色| 亚洲成人精选| 日韩精品欧美在线| 久久九九热re6这里有精品| 国产精品视频精品| 白浆视频在线观看| 久久久国产精品视频| 欧美日韩伦理片| 日韩欧美卡一卡二| japanese国产在线观看| 亚洲va天堂va国产va久| 成年人二级毛片| 久久精品夜夜夜夜久久| 污污免费在线观看| 国产麻豆精品95视频| 狠狠热免费视频| 亚洲人人精品| 自拍偷拍亚洲色图欧美| 久久综合色占| 精品日本一区二区三区在线观看| 国产精品久久久久久久久久久久久久久 | 国产98在线|日韩| 国产情侣一区二区三区| 欧洲成人在线观看| 黑人精品视频| 欧美插天视频在线播放| 福利成人在线观看| 亚洲精品电影久久久| www.成人在线观看| 666欧美在线视频| 中文字幕欧美色图| 色婷婷久久99综合精品jk白丝| www.99re7.com| 一二三区精品视频| 91成人福利视频| 亚洲欧美在线观看| 国产第一页浮力| 国产精品乱码一区二三区小蝌蚪| 免费看污黄网站在线观看| 不卡高清视频专区| 高清中文字幕mv的电影| 国产精品99久久久久久似苏梦涵| 亚洲18在线看污www麻豆 | 1024精品久久久久久久久| 亚洲欧洲在线一区| 四虎成人av| 亚洲欧洲久久| 99久久激情| 性欧美18一19内谢| 中文字幕亚洲综合久久五月天色无吗'' | 五月天亚洲精品| 国产午夜精品无码一区二区| 亚洲自拍偷拍欧美| 日韩精品在线免费看| 午夜一区二区三区视频| 亚洲黄色三级视频| 日韩欧美在线网址| 黄色网址中文字幕| 欧美老女人在线| 国产精品久久久久久久久毛片 | 日韩美女一级视频| 亚洲精品在线观看www| 你懂的在线免费观看| 国产午夜精品免费一区二区三区| 国产网站在线播放| 中文字幕九色91在线| 巨大荫蒂视频欧美另类大| 不卡av日日日| 91黄页在线观看| 91av免费观看91av精品在线| 嗯啊主人调教在线播放视频| 人人澡人人澡人人看欧美| 国产亚洲一区二区手机在线观看| 国产精品久久久久久久久久免费| 91麻豆精品国产91久久久更新资源速度超快| 91在线观看免费| 高清精品xnxxcom| 欧美日韩一区二| 婷婷精品进入| 免费特级黄色片| 久久精选视频| 一女二男3p波多野结衣| 成人丝袜视频网| 一级肉体全黄裸片| 亚洲欧美日韩中文播放| 色网站在线播放| 欧美日韩国产首页| 高清国产mv在线观看| 国产亚洲欧美aaaa| 性欧美videoshd高清| 国产成人一区二区三区电影| 成人在线分类| 久久亚洲免费| 中文字幕一区二区av| 欧美日韩第二页| 国产乱子伦视频一区二区三区 | 自拍偷拍亚洲欧美日韩| 日韩三级av在线| 欧美日韩高清一区二区不卡| 日韩一级免费毛片| 久久激情五月丁香伊人| 新版的欧美在线视频| 亚洲aⅴ男人的天堂在线观看| 日韩动漫一区| 日本美女爱爱视频| 日韩国产高清影视| 国产大学生视频| 中文字幕人成不卡一区| 永久免费无码av网站在线观看| 欧美高清hd18日本| 国产午夜在线观看| 97视频在线观看免费| 精品一区91| 亚洲二区自拍| 亚洲精品麻豆| 日日夜夜精品视频免费观看| 国产午夜一区二区三区| 精品91久久久| 欧美刺激午夜性久久久久久久| 中文字幕日本在线观看| 日韩av免费在线播放| 国产福利一区二区精品秒拍| 超碰成人在线免费观看| 免费精品视频在线| 国产ts在线播放| 色综合久久综合网| 深夜福利免费在线观看| 欧美精品久久久久a| 玖玖玖电影综合影院| 一区二区三区国产福利| 日韩av中文字幕一区二区| 瑟瑟视频在线观看| 懂色av中文一区二区三区天美| 草草视频在线播放| 久久中文字幕在线视频| 天堂综合在线播放| 亚洲一区三区电影在线观看| 视频在线观看一区二区三区| 91精品国产自产| 岛国av在线不卡| 天堂√在线中文官网在线| 91福利视频网| 欧美性生活一级片| 少妇高潮毛片色欲ava片| www.欧美日韩| 成年人免费看毛片| 亚洲高清福利视频| 多野结衣av一区| 免费一区二区三区在在线视频| 999在线观看精品免费不卡网站| 国产污在线观看| 欧美日韩国产在线播放| 日本成人一区二区三区| 国产精品成人免费电影| 精品国产乱码| 五月天视频在线观看| 亚洲视频中文字幕| 国产成人毛毛毛片| 韩国国内大量揄拍精品视频| 麻豆视频一区| 成人性做爰aaa片免费看不忠| 欧美国产国产综合| 国产精品自偷自拍| 九九综合九九综合| 日韩精品福利一区二区三区| 成年人在线看片| 国产精品成人一区二区三区夜夜夜| 在线视频播放大全| 九九精品在线视频| 黄色网一区二区| 福利在线一区二区三区| 国产精品久久久久三级| 国内老熟妇对白xxxxhd| 97福利一区二区| 成人一区二区| 四虎国产精品永久免费观看视频| 午夜亚洲福利老司机| 国产一二三在线观看| 亚洲一区二区中文字幕| 一区二区三区福利| 91av手机在线| 亚洲国产另类久久精品| 日本成人福利| 大陆av在线播放| 国产偷国产偷亚洲高清人白洁| 91中文字幕在线播放| 久久青草福利网站| 日韩国产一区| 人妖粗暴刺激videos呻吟| 色婷婷综合久久久中文字幕| 福利视频在线| 久久人人九九| 国产美女主播视频一区| 亚洲天堂视频网站| 麻豆成人在线看| 猛男gaygay欧美视频| 少妇性l交大片7724com| 91九色最新地址| 久久香蕉av| 中文字幕欧美日韩一区二区| 91蜜桃网址入口| 国产黄色美女视频| 国产精品一区二区久久久| 一区二区国产在线观看|