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

15分鐘手摸手教你寫個(gè)可以操控 Chrome 的插件

開發(fā) 前端 瀏覽器
今天教你寫個(gè)可以操控 Chrome 的插件, 也許這個(gè)插件有許多不完善的地方,主要還是給大家分享個(gè)想法和思路,讓沒接觸過 chrome 插件的朋友們也可以嘗試下

[[411930]]

故事背景

事情是這樣的呢

友人 A: 能不能幫我整一個(gè) chrome 插件?

我: 啥插件?

友人 A: 通過后端服務(wù)或者 python 腳本通信 chrome 插件能夠操作瀏覽器

我: 你小子是想爬數(shù)據(jù)吧?直接用現(xiàn)成的 python 框架或者 谷歌的 puppeteer 就能操控瀏覽器吧

友人 A: 你說的路子我早就試過了,對(duì)于反爬檢測(cè)高的網(wǎng)站一下就能檢測(cè)你的無頭瀏覽器的相應(yīng)特征,所以就用平時(shí)用的瀏覽器就能以真亂真

我: 老是整這些花里胡哨的,有啥用呀

友人 A: 10 斤小龍蝦!

我:成交!!!

整體的思路

根據(jù)朋友以上的要求,我們可以簡單的得出一下的通信流程:

具體有疑問沒關(guān)系,我們只要知道大體的流程是這樣通信的即可

github 地址 每個(gè) commit 對(duì)應(yīng)相應(yīng)的步驟

第一步 創(chuàng)建一個(gè) chrome 插件

我們首先來創(chuàng)建一個(gè)啥功能都沒有的 chrome 插件

目錄如下所示

manifest.json

  1. // manifest.json 
  2.     "manifest_version": 2, // 配置文件的版本 
  3.     "name""SocketEXController", // 插件的名稱 
  4.     "version""1.0.0", // 插件的版本 
  5.     "description""Chrome SocketEXController",// 插件描述 
  6.     "author""wjryours", // 作者 
  7.     "icons": { 
  8.         "48""icon.png",// 對(duì)應(yīng)尺寸的圖標(biāo)路徑 我這邊全部用一個(gè)了 
  9.         "128""icon.png" 
  10.     }, 
  11.     "browser_action": { 
  12.         "default_icon""icon.png", // 圖標(biāo) 
  13.         "default_popup""popup.html" // 點(diǎn)擊右上角的圖標(biāo)的 popup 浮層 html 文件 
  14.     }, 
  15.     "background": { 
  16.         // 會(huì)一直常駐的后臺(tái) JS 或后臺(tái)頁面 
  17.         // 2 種指定方式,如果指定 JS,那么會(huì)自動(dòng)生成一個(gè)背景頁 
  18.         "page""background.html" 
  19.     }, 
  20.     "content_scripts": [ 
  21.         { 
  22.             // 允許哪些域名下加載 注入的 JS 
  23.             // "matches": ["http://*/*""https://*/*"], 
  24.             // "<all_urls>" 表示匹配所有地址 
  25.             "matches": [ 
  26.                 "<all_urls>" 
  27.             ], 
  28.             "js": [ 
  29.                 "content-script.js" 
  30.             ], 
  31.             "run_at""document_start" 
  32.         } 
  33.     ], 
  34.     "permissions": [ 
  35.         "contextMenus", // 右鍵菜單 
  36.         "tabs", // 標(biāo)簽 
  37.         "notifications", // 通知 
  38.         "webRequest", // web 請(qǐng)求 
  39.         "webRequestBlocking", // 阻塞式 web 請(qǐng)求 
  40.         "storage", // 插件本地存儲(chǔ) 
  41.         "http://*/*", // 可以通過 executeScript 或者 insertCSS 訪問的網(wǎng)站 
  42.         "https://*/*" // 可以通過 executeScript 或者 insertCSS 訪問的網(wǎng)站 
  43.     ], 

js

  1. // background.js 
  2. console.log('background.js'
  1. // popup.js 
  2. console.log('popup.js'
  1. // content-script.js 
  2. console.log('content-script.js loaded'

html

  1. <!-- popup --> 
  2. <!DOCTYPE html> 
  3. <html lang="en"
  4. <head> 
  5.     <meta charset="UTF-8"
  6.     <meta http-equiv="X-UA-Compatible" content="IE=edge"
  7.     <meta name="viewport" content="width=device-width, initial-scale=1.0"
  8.     <title>SocketController Popup</title> 
  9.     <link rel="stylesheet" href="./lib/css/popup.css"
  10.     <script src="./popup.js"></script> 
  11. </head> 
  12. <body> 
  13.     popup 
  14. </body> 
  15. </html> 
  1. <!-- background --> 
  2. <!DOCTYPE html> 
  3. <html lang="en"
  4.  
  5. <head> 
  6.     <meta charset="UTF-8"
  7.     <meta http-equiv="X-UA-Compatible" content="IE=edge"
  8.     <meta name="viewport" content="width=device-width, initial-scale=1.0"
  9.     <title>SocketController</title> 
  10. </head> 
  11.  
  12. <body> 
  13.     <div class="bg-container"
  14.         bg-container 
  15.     </div> 
  16. </body> 
  17.  
  18. </html> 

然后在 chrome 的擴(kuò)展程序頁加載我們的文件目錄 即可

然后我們啟用插件 隨手打開一個(gè)頁面就發(fā)現(xiàn)我們的插件已經(jīng)生效了

第二步 在本地創(chuàng)建 websocket 的服務(wù)

正如上面的通信流程所示,我們還需要在本地創(chuàng)建一個(gè)可用的 websocket 來發(fā)送信息給 chrome 插件

為了方便起見,我這邊就用 node 的 express 以及 socket.io 這個(gè)庫來啟用

目錄結(jié)構(gòu)和代碼都很簡單

  1. // index.js  用來創(chuàng)建 node 服務(wù) 
  2. const express = require('express'
  3. const app = express() 
  4. const http = require('http'
  5. const server = http.createServer(app) 
  6. const { Server } = require("socket.io"
  7. const io = new Server(server) 
  8.  
  9. app.get('/', (req, res) => { 
  10.     res.sendFile(__dirname + '/index.html'
  11. }) 
  12.  
  13. io.on('connection', (socket) => { 
  14.     console.log('a user connected'
  15.     socket.on('disconnect', () => { 
  16.         console.log('user disconnected'); 
  17.     }); 
  18.     socket.on('webviewEvent', (msg) => { 
  19.         console.log('webviewEvent: ' + msg); 
  20.         io.emit('webviewEvent', msg); 
  21.         // socket.broadcast.emit('chat message', msg); 
  22.     }); 
  23.     socket.on('webviewEventCallback', (msg) => { 
  24.         console.log('webviewEventCallback: ' + msg); 
  25.         io.emit('webviewEventCallback', msg); 
  26.     }); 
  27. }) 
  28.  
  29.  
  30. server.listen(9527, () => { 
  31.     console.log('listening on 9527'
  32. }) 
  1. <!-- index.html -->  
  2. <!-- 點(diǎn)擊事件傳遞的參數(shù)后續(xù)會(huì)用到,這里可以不去了解 --> 
  3. <!DOCTYPE html> 
  4. <html> 
  5.  
  6. <head> 
  7.   <title>Socket.IO Page</title> 
  8.   <style> 
  9. </head> 
  10.  
  11. <body> 
  12.   <input id="SendInput" autocomplete="off" /> 
  13.   <button id="SendInputevent">Send input event</button> 
  14.   <button id="SendClickevent">Send click event</button> 
  15.   <button id="SendGetTextevent">Send getText event</button> 
  16. </body> 
  17. <script src="/socket.io/socket.io.js"></script> 
  18. <script> 
  19.   var socket = io(); 
  20.  
  21.   var form = document.getElementById('form'); 
  22.   var input = document.getElementById('input'); 
  23.  
  24.   document.getElementById('SendClickevent').addEventListener('click'function (e) { 
  25.     socket.emit('webviewEvent', { event: 'click', params: { delay: 300 }, element: '#su', operateTabIndex: 0 }); 
  26.   }) 
  27.   document.getElementById('SendInputevent').addEventListener('click'function (e) { 
  28.     const value = document.getElementById('SendInput').value 
  29.     socket.emit('webviewEvent', { event: 'input', params: { inputValue: value }, element: '#kw', operateTabIndex: 0 }); 
  30.   }) 
  31.   document.getElementById('SendGetTextevent').addEventListener('click'function (e) { 
  32.     socket.emit('webviewEvent', { event: 'getElementText', params: {}, element: '.result.c-container.new-pmd .t a', operateTabIndex: 0 }); 
  33.   }) 
  34.  
  35.   socket.on('webviewEventCallback', (msg) => { 
  36.     console.log(msg) 
  37.   }) 
  38. </script> 
  39.  
  40. </html> 
  1. // package.json 
  2.   "name""socket-service"
  3.   "version""1.0.0"
  4.   "description"""
  5.   "main""index.js"
  6.   "scripts": { 
  7.     "test""echo \"Error: no test specified\" && exit 1"
  8.     "dev""nodemon index.js" 
  9.   }, 
  10.   "author"""
  11.   "license""ISC"
  12.   "dependencies": { 
  13.     "express""^4.17.1"
  14.     "nodemon""^2.0.7"
  15.     "socket.io""^4.1.2" 
  16.   } 

具體的內(nèi)容也很簡單,就是使用 express 和 socket.io 創(chuàng)建了一個(gè) node 服務(wù)支持長鏈接,對(duì)于 socket.io 想有更多的了解的可以參照 官方文檔

運(yùn)行 npm run dev 即可

好的,這樣我們的服務(wù)就跑起來了

我們?cè)L問 http://localhost:9527

并點(diǎn)擊頁面上的按鈕在命令行上有 log 輸出就說明連接成功啦!

第三步 開始使 chrome 插件 與 本地的 node 服務(wù)相互通信

在開始與 node 服務(wù)通信前我們要了解下 chrome 插件的幾種 js 的使用場(chǎng)景

content-scripts

這個(gè)主要功能就是 Chrome 插件中向頁面注入腳本 在第一步的操作中正是該文件在別的頁面控制臺(tái)中打印出了我們期望的 log content-scripts 和 原始頁面共享 DOM,但是不共享 JS 但是這個(gè)功能足以讓我們?nèi)ゲ僮髂繕?biāo)頁面了

background.js

是一個(gè)常駐的頁面,它的生命周期是插件中所有類型頁面中最長的,它隨著瀏覽器的打開而打開, 隨著瀏覽器的關(guān)閉而關(guān)閉,所以通常把需要一直運(yùn)行的、啟動(dòng)就運(yùn)行的、全局的代碼放在 background 里面

popup.js

這個(gè)就是點(diǎn)擊瀏覽器右上角的插件圖標(biāo)展示的彈窗,生命周期很短,可以將臨時(shí)的交互寫在這里

對(duì)于我們這次要長時(shí)間駐存在瀏覽器后臺(tái)與服務(wù)通信的要求得出 我們將相應(yīng)的寫在 background.js 中即可

我們這里將需要的 js 庫 和 background.js 引入到 background.html 中

  1. <script src="./lib/js/lodash.min.js"></script> 
  2. <script src="./lib/js/socket.io.min.js"></script> 
  3. <script src="./background.js"></script> 

我們可以使用兩種方式來調(diào)試 這個(gè)常駐后臺(tái)文件

1.直接在 chrome 拓展點(diǎn)擊對(duì)應(yīng)按鈕即可彈出調(diào)試

2.直接在瀏覽器上輸入對(duì)應(yīng)的地址 即可

  1. chrome-extension://${extensionID}/background.html 

每次更新代碼點(diǎn)擊按鈕刷新即可

為了調(diào)試方便起見我在 popup.js 中加入了以下代碼 每次點(diǎn)擊我們的插件圖標(biāo)即可新開一個(gè)后臺(tái)頁面

  1. const extensionId = chrome.runtime.id 
  2. const backgroundURL = `chrome-extension://${extensionId}/background.html` 
  3. window.open(backgroundURL) 

現(xiàn)在我們只需要在 background.js 中編寫相應(yīng)代碼,建立長鏈接就可以了

  1. // background.js 
  2. class BackgroundService { 
  3.     constructor() { 
  4.         this.socketIoURL = 'http://localhost:9527' 
  5.         this.socketInstance = {} 
  6.         this.socketRetryMax = 5 
  7.         this.socketRetry = 0 
  8.     } 
  9.     init() { 
  10.         console.log('background.js')    
  11.         this.connectSocket() 
  12.         this.linstenSocketEvent() 
  13.     } 
  14.     setSocketURL(url) { 
  15.         this.socketIoURL = url 
  16.     } 
  17.     connectSocket() { 
  18.         if (!_.isEmpty(this.socketInstance) && _.isFunction(this.socketInstance.disconnect)) { 
  19.             this.socketInstance.disconnect() 
  20.         } 
  21.         this.socketInstance = io(this.socketIoURL); 
  22.         this.socketRetry = 0 
  23.         this.socketInstance.on('connect_error', (e) => { 
  24.             console.log('connect_error', e) 
  25.             this.socketRetry++ 
  26.             if (this.socketRetryMax < this.socketRetry) { 
  27.                 this.socketInstance.close() 
  28.                 alert(`以嘗試連接${this.socketRetryMax}次,無法連接到 socket 服務(wù),請(qǐng)排查服務(wù)是否可用`) 
  29.             } 
  30.         }) 
  31.     } 
  32.     linstenSocketEvent() { 
  33.         if (!_.isEmpty(this.socketInstance) && _.isFunction(this.socketInstance.on)) { 
  34.             this.socketInstance.on('webviewEvent', (msg) => { 
  35.                 console.log(`webviewEvent msg`, msg) 
  36.             }); 
  37.         } 
  38.     } 
  39. const app = new BackgroundService() 
  40. app.init() 

刷新插件,打開插件后臺(tái)頁面 就可以看見鏈接建立成功,然后從 node 服務(wù)發(fā)送 msg 給 chrome 插件,我們就可以看到信息被成功接收了

(tips:之前的 node 服務(wù)別忘記啟動(dòng))

第四步 開始使 chrome 插件 background.js 與 content-script.js 建立通信

這一步也是相當(dāng)簡單,chrome 官方的文檔也有很多介紹 我這邊就寫下實(shí)現(xiàn)步驟

  1. // 修改 background.js 為如下代碼 
  2. static emitMessageToSocketService(socketInstance, params = {}) { 
  3.     if (!_.isEmpty(socketInstance) && _.isFunction(socketInstance.emit)) { 
  4.         console.log(params) 
  5.         // 將從 content-script.js 接收到的 msg 發(fā)送到 node 服務(wù) 
  6.         socketInstance.emit('webviewEventCallback', params); 
  7.     } 
  8. linstenSocketEvent() { 
  9.     if (!_.isEmpty(this.socketInstance) && _.isFunction(this.socketInstance.on)) { 
  10.         this.socketInstance.on('webviewEvent', (msg) => { 
  11.             console.log(`webviewEvent msg`, msg) 
  12.             // 將從 node 服務(wù)接收到的 msg 發(fā)送到 content-script.js 
  13.             this.sendMessageToContentScript(msg, BackgroundService.emitMessageToSocketService) 
  14.         }); 
  15.     } 
  16. sendMessageToContentScript(message, callback) { 
  17.     const operateTabIndex = message.operateTabIndex ? message.operateTabIndex : 0 
  18.     console.log(message) 
  19.     chrome.tabs.query({ index: operateTabIndex }, (tabs) => { // 獲取 索引的方式獲取對(duì)應(yīng) tabs 實(shí)例以及 id 
  20.         chrome.tabs.sendMessage(tabs[0].id, message, (response) => { // 發(fā)送消息到對(duì)應(yīng) tab 
  21.             console.log(callback) 
  22.             if (callback) callback(this.socketInstance, response) 
  23.         }); 
  24.     }); 
  1. // content-script.js 
  2.  
  3. chrome.runtime.onMessage.addListener(function (request, sender, sendResponse) { 
  4.     console.log(request, sender, sendResponse) 
  5.     sendResponse(res) 
  6. }); 

然后我們這邊將插件重新加載后關(guān)閉瀏覽器重新打開新瀏覽器,將需要測(cè)試的頁面放置在第一個(gè), 然后在我們的 localhost:9527 發(fā)送信息 這是我們就能在我們預(yù)期的頁面接收到對(duì)應(yīng)參數(shù)了

 

這時(shí)你可能會(huì)看到 2 條 log,其實(shí)這個(gè)是正常現(xiàn)象, 因?yàn)槿绻闶峭ㄟ^打開了 chrome-extension://xxx/background.html 直接打開后臺(tái)頁 運(yùn)行一個(gè)后臺(tái)線程 但是真正在后臺(tái)常駐的還有一個(gè)線程 所以相當(dāng)是 2 個(gè)后臺(tái)接收到了 socket 消息所以就發(fā)送 2 次 msg

第五步 嘗試操控瀏覽器做對(duì)應(yīng)操作

好的,朋友們,我們終于來到了最后一步了

我們現(xiàn)在已經(jīng)建立起了這 3 個(gè)模塊間的聯(lián)系了 現(xiàn)在無非就是要將從后端發(fā)送的消息通過一些判斷做一些 js 操作

我們就來完成一個(gè)簡單的任務(wù),打開百度頁面,搜索關(guān)鍵字,并將搜索到的各個(gè) title 獲取

我這邊為了做演示方便點(diǎn)就直接引入了 jq 來操作 dom 在 js 文件夾下創(chuàng)建 operate.js 以及 jquery.min.js

  1. // 在 manifest.json 中加入 相應(yīng) js 
  2. "content_scripts": [ 
  3.     { 
  4.         "matches": [ 
  5.             "<all_urls>" 
  6.         ], 
  7.         "js": [ 
  8.             "lib/js/jquery.min.js"
  9.             "lib/js/operate.js"
  10.             "content-script.js" 
  11.         ], 
  12.         "run_at""document_start" 
  13.     } 

operate.js 主要用來定義一些操作

根據(jù)我們上面的小任務(wù),我這邊現(xiàn)在這里面加幾個(gè)簡單的事件定義,后續(xù)可以支持?jǐn)U展

  1. // operate.js 
  2. const operateTypeMap = { 
  3.     CLICK: 'click'
  4.     INPUT: 'input'
  5.     GETELEMENTTEXT: 'getElementText' 
  6.  
  7. class OperateConstant { 
  8.     static operateByEventType(type, payload = {}) { 
  9.         let res 
  10.         switch (type) { 
  11.             case operateTypeMap.CLICK: 
  12.                 res = OperateConstant.handleClickEvent(payload) 
  13.                 break; 
  14.             case operateTypeMap.INPUT: 
  15.                 res = OperateConstant.handleInputEvent(payload) 
  16.                 break; 
  17.             case operateTypeMap.GETELEMENTTEXT: 
  18.                 res = OperateConstant.handleGetElementTextEvent(payload) 
  19.                 break; 
  20.             default
  21.                 break; 
  22.         } 
  23.         return res 
  24.     } 
  25.     static handleClickEvent(payload) { 
  26.         let data = null 
  27.         if (payload.element) { 
  28.             $(payload.element).click() 
  29.         } 
  30.         return data 
  31.     } 
  32.     static handleInputEvent(payload) { 
  33.         let data = null 
  34.         if (payload.element) { 
  35.             $(payload.element).val(payload.params.inputValue) 
  36.         } 
  37.         return data 
  38.     } 
  39.     static handleGetElementTextEvent(payload) { 
  40.         let data = [] 
  41.         if (payload.element && $(payload.element)) { 
  42.             Array.from($(payload.element)).forEach((item) => { 
  43.                 const resItem = { 
  44.                     value: $(item).text() 
  45.                 } 
  46.                 data.push(resItem) 
  47.             }) 
  48.         } 
  49.         return data 
  50.     } 

然后在 conent-script.js 使用

  1. chrome.runtime.onMessage.addListener(function (request, sender, sendResponse) { 
  2.     const operateRes =  OperateConstant.operateByEventType(request.event, request) 
  3.     console.log(operateRes) 
  4.     const res = { 
  5.         code: 0, 
  6.         data: operateRes, 
  7.         message: '操作成功' 
  8.     } 
  9.     sendResponse(res) 
  10. }); 

好的,我們來試下我們的功能吧 (tips: 請(qǐng)重新加載插件關(guān)閉所有 tab 以及確保你想要測(cè)試的 tabs 處于第一個(gè))

可以,非常完美

小結(jié)

好的,朋友們,今天的分享就到這里了, 也許這個(gè)插件有許多不完善的地方,主要還是給大家分享個(gè)想法和思路,讓沒接觸過 chrome 插件的朋友們也可以嘗試下

 

責(zé)任編輯:姜華 來源: 微醫(yī)大前端技術(shù)
相關(guān)推薦

2022-09-06 08:40:33

應(yīng)用系統(tǒng)登錄方式Spring

2022-06-17 08:05:28

Grafana監(jiān)控儀表盤系統(tǒng)

2013-06-27 09:41:19

LuaLua語言Lua語言快速入門

2014-04-22 09:42:12

Bash腳本教程

2020-10-19 18:07:00

云計(jì)算技術(shù)應(yīng)用

2021-08-01 21:38:07

網(wǎng)頁點(diǎn)燈網(wǎng)關(guān)

2018-04-24 14:52:48

LinuxBash腳本

2017-10-11 15:17:42

sklearn機(jī)器學(xué)習(xí)pandas

2018-08-30 14:31:28

Linux磁盤LVM

2018-04-20 16:43:23

2015-07-08 09:43:22

程序員

2022-01-26 00:02:00

Nacos服務(wù)注冊(cè)中心

2019-01-16 18:34:37

Python 開發(fā)數(shù)據(jù)

2009-10-09 14:45:29

VB程序

2020-01-22 16:40:48

Java開發(fā)代碼

2021-10-27 05:47:53

通信協(xié)議協(xié)議網(wǎng)絡(luò)技術(shù)

2018-01-16 10:11:11

Nginx訪問日志

2022-09-30 15:46:26

Babel編譯器插件

2025-10-16 10:00:00

運(yùn)維vim編輯器

2015-08-06 17:17:33

swoole聊天室
點(diǎn)贊
收藏

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

亚洲加勒比久久88色综合| 亚洲国产精品久久一线不卡| 国产精品偷伦免费视频观看的| 国产午夜精品全部视频在线播放| 成人综合日日夜夜| 亚洲美女在线一区| 蜜桃av色综合| 国产特级黄色片| 久久精品伊人| 欧美裸体男粗大视频在线观看| 黄色短视频在线观看| 亚洲综合伊人| 欧美午夜精品久久久久久久| a级网站在线观看| 九色在线观看视频| 国产精品一区专区| 国产精品va在线| 精品无码黑人又粗又大又长| 日韩国产一区二区| 日韩av中文字幕在线播放| 91女神在线观看| 中文字幕在线官网| 亚洲国产一区视频| 中文字幕av日韩精品| 免费av在线电影| 波多野结衣在线一区| 91视频免费在线| 激情网站在线观看| 国产精品日本欧美一区二区三区| 蜜月aⅴ免费一区二区三区| 在线视频第一页| 欧美做受69| 精品成人一区二区三区| 久久精品无码一区二区三区毛片| 电影亚洲一区| 色欧美日韩亚洲| 3d动漫一区二区三区| 久草在线资源站资源站| 亚洲欧美一区二区三区久本道91 | 久久一区二区三区视频| 欧美日本一区二区视频在线观看 | 国产成人在线免费| 亚洲xxxxx| 国产精品久久久久久久免费看| 天堂在线一区二区| 欧美综合激情网| 国产精品黄色网| 亚洲美女网站| 97国产精品视频| 久久精品免费在线| 黄色亚洲精品| 97在线观看免费高清| 国产无遮挡裸体免费视频| 欧美涩涩视频| 久久久亚洲精品视频| 国产大片中文字幕在线观看| 红桃视频欧美| 超在线视频97| 免费看一级一片| 黄色欧美成人| 欧美在线不卡区| 日日夜夜操视频| 日本不卡视频一二三区| 国产精品视频免费在线观看| 国产一区二区三区三州| 国内成人自拍视频| 成人av免费看| 手机看片福利在线观看| 久久久99精品免费观看| 欧美久久久久久一卡四| h视频网站在线观看| 国产精品久久久久久亚洲伦| 亚洲成人动漫在线| 黑森林国产精品av| 天天色天天操综合| 久久久国产欧美| 成人51免费| 亚洲精品福利视频| 中文字幕免费高清| 欧美在线黄色| 91精品国产乱码久久久久久久久| 日韩精品成人免费观看视频| 精品无码三级在线观看视频 | 噜噜噜久久,亚洲精品国产品| 99精品视频中文字幕| 日韩精品另类天天更新| 在线午夜影院| 色哟哟亚洲精品| 欧美午夜精品理论片| 国产女人18毛片水真多18精品| 精品在线小视频| 97在线观看视频免费| 最新国产拍偷乱拍精品| 国产精品美腿一区在线看| www.久久成人| 国产欧美日韩麻豆91| 亚洲色婷婷久久精品av蜜桃| 成人小电影网站| 91精品国产色综合久久ai换脸| 在线黄色免费网站| 欧美xxxxx视频| 欧美有码在线观看| www.日日夜夜| 国产精品视频第一区| 高清欧美精品xxxxx| 九七影院97影院理论片久久 | 欧美性猛交xxxx免费看| 亚洲高清视频免费| 精品中文一区| 欧美激情精品久久久久久蜜臀 | 日韩三级视频在线看| 亚洲久久久久久久| 亚洲欧洲一区| 96国产粉嫩美女| 成人在线观看免费| 狠狠躁夜夜躁人人躁婷婷91| 少妇性l交大片7724com| 成人一区二区| 欧美在线不卡区| 天堂成人在线视频| 亚洲欧美激情在线| 手机版av在线| 全球成人免费直播| 日韩美女福利视频| 污污视频在线观看网站| 亚洲资源在线观看| 色偷偷中文字幕| 久久精品高清| 国产精品69av| 青春草在线观看| 天天综合网 天天综合色| 黑人巨大猛交丰满少妇| 五月天久久久| 国产情人节一区| 69久久精品| 欧美天堂一区二区三区| 中文字幕免费视频| 久久久久久夜| 日韩福利影院| 久久亚洲精品爱爱| 亚洲午夜小视频| 成人av网站在线播放| 91啦中文在线观看| 国产 福利 在线| 日韩动漫一区| 18一19gay欧美视频网站| 亚洲精品久久久久久动漫器材一区| 综合在线观看色| 亚洲黄色片免费看| 欧美成人69| 国产v亚洲v天堂无码| 好久没做在线观看| 亚洲国产精品999| 99精品视频99| 成人av在线影院| 欧美精品一区免费| 欧美精品一区二区久久| 国产精品色婷婷视频| 色影视在线观看| 欧美一激情一区二区三区| 亚洲国产美女视频| 成人动漫一区二区| 北条麻妃69av| 欧美综合在线视频观看| 成人字幕网zmw| 亚洲小说区图片| 亚洲国产小视频| 中文字幕精品视频在线观看| 中文一区一区三区高中清不卡| 日本高清久久久| 国自产拍偷拍福利精品免费一| 国产一区高清视频| 国产精品毛片久久久久久久久久99999999| 日韩中文av在线| 超碰免费在线97| 欧美色播在线播放| 一本在线免费视频| 成人黄色大片在线观看 | 亚洲天堂色网站| 中文字幕在线播出| 亚洲国产视频一区| 精品国产成人亚洲午夜福利| 久久99精品国产91久久来源| 日韩国产小视频| 国产一区国产二区国产三区| 亚洲jizzjizz日本少妇| 色网在线免费观看| 日韩视频欧美视频| 天天影院图片亚洲| 91精品国产色综合久久| 日韩在线视频免费播放| 亚洲天堂精品在线观看| chinese麻豆新拍video| 韩国成人福利片在线播放| 成人性生活视频免费看| 日韩成人精品一区| 精品国产免费一区二区三区| 欧美黑粗硬大| 啪一啪鲁一鲁2019在线视频| 国产原厂视频在线观看| 亚洲欧美日韩天堂| 亚洲AV无码精品国产| 欧美视频三区在线播放| 精品视频一区二区在线观看| 中文字幕视频一区| 国产精品扒开腿做爽爽| 国产成人鲁色资源国产91色综| 欧美日韩怡红院| 午夜激情一区| 亚洲午夜精品久久| 亚洲图片久久| 成人av网站观看| 四虎影视精品永久在线观看| 日韩美女主播视频| 7777kkk亚洲综合欧美网站| 精品国产一区av| 精品视频一二区| 日韩精品极品视频| 狠狠人妻久久久久久综合麻豆| 91麻豆精品国产91久久久使用方法| 欧美日韩一级黄色片| 亚洲成人免费视频| 久久免费精彩视频| 亚洲精品乱码久久久久久久久| 黄色av片三级三级三级免费看| 久久久国产午夜精品| 久久精品国产亚洲av麻豆| 成人午夜av在线| 又大又长粗又爽又黄少妇视频| 久草在线在线精品观看| 亚洲无吗一区二区三区| 久久久久久9| 无码人妻精品一区二区三区66| 一本久久综合| 国产精品12345| 亚洲区欧美区| www插插插无码免费视频网站| 亚洲精品成人影院| 黄色一级片网址| 亚洲国产日韩欧美在线| 日本丰满少妇黄大片在线观看| 91久久夜色精品国产按摩| 日韩在线第一区| 成人羞羞视频在线看网址| 翔田千里亚洲一二三区| 亚洲动漫精品| 日韩av电影免费在线观看| 欧美精美视频| 日韩精品久久一区| 日韩欧美精品一区| 一级黄色录像免费看| 亚洲国产精品久久久天堂| 欧美交换配乱吟粗大25p| 激情综合在线| 欧美s码亚洲码精品m码| 老牛国产精品一区的观看方式| av网址在线观看免费| 日韩av网站免费在线| 不卡的在线视频| 国产福利视频一区二区三区| 日本国产在线视频| 久久婷婷色综合| 日韩欧美黄色网址| 亚洲三级电影网站| 日韩精品成人一区| 黑人巨大精品欧美一区二区三区 | 日韩高清a**址| 国产综合在线观看| www.欧美三级电影.com| 污污的网站在线免费观看| 性欧美激情精品| 欧美xxxx做受欧美护士| 成人免费视频网| 加勒比色老久久爱综合网| 欧美日韩最好看的视频| 国产精品成人一区二区不卡| 99热这里只有精品免费| 国产美女精品| 超碰成人在线播放| 成人激情小说网站| 久久久久久成人网| 亚洲最新视频在线观看| 国产一级一级国产| 欧美一区二区三区四区高清| 亚洲色偷精品一区二区三区| 日日摸夜夜添一区| av资源在线播放| 国产精品综合久久久| 91夜夜蜜桃臀一区二区三区| 奇米视频888战线精品播放| 欧美成人日本| 一区二区三区免费播放| 成人精品国产一区二区4080| 99久久99久久精品免费看小说. | 青青在线免费观看| 日本vs亚洲vs韩国一区三区| 无码人妻一区二区三区一| 欧美国产日韩a欧美在线观看| 久久精品99久久久久久| 欧美日韩国产美| 日韩偷拍自拍| 久久99精品久久久久久琪琪| 51一区二区三区| 精品国产乱码久久久久久郑州公司| 先锋资源久久| 蜜臀久久99精品久久久酒店新书| 国产高清亚洲一区| 老司机福利在线观看| 日韩欧美亚洲一二三区| 国产1区在线观看| 成年无码av片在线| 国产精品天堂蜜av在线播放| 久久久久成人精品免费播放动漫| 99久久精品网| 久久婷婷综合色| 久久久99久久| 亚洲va在线观看| 亚洲国产成人久久综合| 日本无删减在线| 成人福利视频在线观看| 精品一区二区三区中文字幕老牛| 国产精品999视频| 成人污污视频在线观看| 久久国产波多野结衣| 欧美色图免费看| 精品视频一二区| 国产极品jizzhd欧美| 久草成人在线| 情侣黄网站免费看| 91美女片黄在线观看91美女| 国产精品99无码一区二区| 日韩欧美国产系列| 日本天码aⅴ片在线电影网站| 91久久大香伊蕉在人线| 天天做天天爱天天综合网2021 | 成人a在线观看高清电影| 欧美日韩精品免费观看| 国产午夜精品一区二区三区欧美| 李丽珍裸体午夜理伦片| 亚洲国产日韩综合久久精品| 亚洲第九十九页| 欧美激情视频三区| 88久久精品| 国产女主播自拍| 97精品国产露脸对白| 国产无遮挡免费视频| 亚洲黄色有码视频| 日本在线播放一二三区| 鲁片一区二区三区| 久久国产88| 欧美一区二区三区粗大| 欧美久久婷婷综合色| 成人在线播放免费观看| 91美女片黄在线观| 欧美午夜久久| 手机在线看片日韩| 欧美在线一区二区三区| 蜜桃av在线免费观看| 亚洲一区二区久久久久久久| 国产中文一区| 天天插天天射天天干| 欧美优质美女网站| 求av网址在线观看| ts人妖另类在线| 午夜亚洲激情| 国产性猛交xx乱| 91精品国产欧美一区二区| 午夜影院免费在线| 久久国产一区二区| 久久精品999| 国产精品theporn动漫| 亚洲人成77777在线观看网| 亚洲精品69| 自慰无码一区二区三区| 亚洲国产精品成人久久综合一区 | 日韩精品极品在线观看| 欧美色网在线| 国产91视频一区| 久久久久久久精| ,一级淫片a看免费| 97精品视频在线观看| 91日韩免费| 中文字幕日韩三级片| 欧美日韩精品免费| av在线视屏| 最新国产精品久久| av亚洲精华国产精华| 一级黄色片视频| 欧美亚洲在线播放| 亚洲一区在线| 国产ts在线播放| 亚洲精品在线观看网站| 亚洲精品555| 好吊妞无缓冲视频观看| 亚洲欧美日韩综合aⅴ视频| 日本人妖在线| y111111国产精品久久婷婷| 日韩精品一级二级| 日韩美女黄色片| 久久久97精品|