国产熟女一区二区91,人妻少妇精品久久久久久蜜臀av,99re国产在线观看,婷婷午夜av天堂,日本东京热久久久a,日本欧美一级二级三级,欧美另类丝袜一区在线,国产精品白浆久久久久久久,久99re在线播放

【由淺入深】vue組件庫實戰(zhàn)開發(fā)總結分享-天天熱聞

來源:php中文網 | 2022-12-27 20:01:38 |

很慶幸標題能夠趕上2022結束的腳步。本文由淺入深層層遞進,對組件庫的開發(fā)過程做個了小結。

由于篇幅有限,陰影部分的內容將在中/下篇介紹。

話不多說,直入主題。


【資料圖】

yarn workspace + lerna: 管理組件庫及其生態(tài)項目

考慮到組件庫整體需要有多邊資源支持,比如組件源碼,庫文檔站點,color-gen等類庫工具,代碼規(guī)范配置,vite插件,腳手架,storybook等等,需要分出很多packages,package之間存在彼此聯(lián)系,因此考慮使用monorepo的管理方式,同時使用yarn作為包管理工具,lerna作為包發(fā)布工具?!鞠嚓P推薦:vuejs視頻教程、web前端開發(fā)】

在monorepo之前,根目錄就是一個workspace,我們直接通過yarn add/remove/run等就可以對包進行管理。但在monorepo項目中,根目錄下存在多個子包,yarn 命令無法直接操作子包,比如根目錄下無法通過yarn run dev啟動子包package-a中的dev命令,這時我們就需要開啟yarn的workspaces功能,每個子包對應一個workspace,之后我們就可以通過yarn workspace package-a run dev啟動package-a中的dev命令了。

你可能會想,我們直接cd到package-a下運行就可以了,不錯,但yarn workspaces的用武之地并不只此,像auto link,依賴提升,單.lock等才是它在monorepo中的價值所在。

啟用yarn workspaces

我們在根目錄packge.json中啟用yarn workspaces:

{  "private": true,  "workspaces": [    "packages/*"  ]}

packages目錄下的每個直接子目錄作為一個workspace。由于我們的根項目是不需要發(fā)布出去的,因此設置private為true。

安裝lerna并初始化

不得不說,yarn workspaces已經具備了lerna部分功能,之所以使用它,是想借用它的發(fā)布工作流以彌補workspaces在monorepo下在這方面的不足。下面我們開始將lerna集成到項目中。

首先我們先安裝一下lerna:

# W指workspace-root,即在項目根目錄下安裝,下同yarn add lerna -D -W# 由于經常使用lerna命令也推薦全局安裝yarn global add lernaornpm i lerna -g

執(zhí)行lerna init初始化項目,成功之后會幫我們創(chuàng)建了一個lerna.json文件

lerna init
// lerna.json{  "$schema": "node_modules/lerna/schemas/lerna-schema.json",  "useWorkspaces": true,  "version": "0.0.0"}

$schema指向的lerna-schema.json描述了如何配置lerna.json,配置此字段后,鼠標懸浮在屬性上會有對應的描述。注意,以上的路徑值需要你在項目根目錄下安裝lerna。

useWorkspaces定義了在lerna bootstrap期間是否結合yarn workspace。

由于lerna默認的工作模式是固定模式,即發(fā)布時每個包的版本號一致。這里我們修改為independent獨立模式,同時將npm客戶端設置為yarn。如果你喜歡pnpm,just do it!

// lerna.json{  "version": "independent",  "npmClient": "yarn"}

至此yarn workspaces搭配lerna的monorepo項目就配置好了,非常簡單!

額外的lerna配置

By the way!由于項目會使用commitlint對提交信息進行校驗是否符合Argular規(guī)范,而lerna version默認為我們commit的信息是"Publish",因此我們需要進行一些額外的配置。

// lerna.json{  "command": {    "version": {      "message": "chore(release): publish",      "conventionalCommits": true    }  }}

可以看到,我們使用符合Argular團隊提交規(guī)范的"chore(release): publish"代替默認的"Publish"。

conventionalCommits表示當我們運行lerna version,實際上會運行lerna version --conventional-commits幫助我們生成CHANGELOG.md。

小結

在lerna剛發(fā)布的時候,那時的包管理工具還沒有可用的workspaces解決方案,因此lerna自身實現(xiàn)了一套解決方案。時至今日,現(xiàn)代的包管理工具幾乎都內置了workspaces功能,這使得lerna和yarn有許多功能重疊,比如執(zhí)行包pkg-a的dev命令lerna run dev --stream --scope=pkg-a,我們完全可以使用yarn workspace pkg-a run dev代替。lerna bootstrap --hoist將安裝包提升到根目錄,而在yarn workspaces中直接運行yarn就可以了。

Anyway, 使用yarn作為軟件包管理工具,lerna作為軟件包發(fā)布工具,是在monorepo管理方式下一個不錯的實踐!

集成Lint工具規(guī)范化代碼

很無奈,我知道大部分人都不喜歡Lint,但對我而言,這是必須的。

集成eslint

packages目錄下創(chuàng)建名為@argo-design/eslint-config(非文件夾名)的package

1. 安裝eslint

cd argo-eslint-configyarn add eslintnpx eslint --init

注意這里沒有-D或者--save-dev。選擇如下:

安裝完成后手動將devDependencies下的依賴拷貝到dependencies中?;蛘吣闶謩影惭b這一系列依賴。

2. 使用

// argo-eslint-config/package.json{  scripts: {    "lint:script": "npx eslint --ext .js,.jsx,.ts,.tsx --fix --quiet ./"  }}

運行yarn lint:script,將會自動修復代碼規(guī)范錯誤警告(如果可以的話)。

3. VSCode保存時自動修復

安裝VSCode Eslint插件并進行如下配置,此時在你保存代碼時,也會自動修復代碼規(guī)范錯誤警告。

// settings.json{  "editor.defaultFormatter": "dbaeumer.vscode-eslint",  "editor.codeActionsOnSave": {    "source.fixAll.eslint": true  }}

4. 集成到項目全局

argo-eslint-config中新建包入口文件index.js,并將.eslintrc.js的內容拷貝到index.js中

module.exports = {  env: {    browser: true,    es2021: true,    node: true  },  extends: ["plugin:vue/vue3-essential", "standard-with-typescript"],  overrides: [],  parserOptions: {    ecmaVersion: "latest",    sourceType: "module"  },  plugins: ["vue"],  rules: {}}

確保package.json配置main指向我們剛剛創(chuàng)建的index.js。

// argo-eslint-config/package.json{   "main": "index.js"}

根目錄package.json新增如下配置

// argo-eslint-config/package.json{  "devDependencies": {    "@argo-design/eslint-config": "^1.0.0"  },  "eslintConfig": {    "root": true,    "extends": [      "@argo-design"    ]  }}

最后運行yarn重新安裝依賴。

注意包命名與extends書寫規(guī)則;root表示根配置,對eslint配置文件冒泡查找到此為止。

集成prettier

接下來我們引入formatter工具prettier。首先我們需要關閉eslint規(guī)則中那些與prettier沖突或者不必要的規(guī)則,最后由prettier代為實現(xiàn)這些規(guī)則。前者我們通過eslint-config-prettier實現(xiàn),后者借助插件eslint-plugin-prettier實現(xiàn)。比如沖突規(guī)則尾逗號,eslint-config-prettier幫我們屏蔽了與之沖突的eslint規(guī)則:

{  "comma-dangle": "off",  "no-comma-dangle": "off",  "@typescript-eslint/comma-dangle": "off",  "vue/comma-dangle": "off",}

通過配置eslint規(guī)則"prettier/prettier": "error"讓錯誤暴露出來,這些錯誤交給eslint-plugin-prettier收拾。

prettier配置我們也新建一個package@argo-design/prettier-config。

1. 安裝

cd argo-prettier-configyarn add prettieryarn add eslint-config-prettier eslint-plugin-prettier

2. 使用

// argo-prettier-config/index.jsmodule.exports = {  printWidth: 80, //一行的字符數(shù),如果超過會進行換行,默認為80  semi: false, // 行尾是否使用分號,默認為true  trailingComma: "none", // 是否使用尾逗號  bracketSpacing: true // 對象大括號直接是否有空格};

完整配置參考官網 prettier配置

3. 配置eslint

回到argo-eslint-config/index.js,只需新增如下一條配置即可

module.exports = {   "extends": ["plugin:prettier/recommended"]};

plugin:prettier/recommended指的eslint-plugin-prettierpackage下的recommended.js。該擴展已經幫我們配置好了

{  "extends": ["eslint-config-prettier"],  "plugins": ["eslint-plugin-prettier"],  "rules": {    "prettier/prettier": "error",    "arrow-body-style": "off",    "prefer-arrow-callback": "off"  }}

4. 集成到項目全局

根目錄package.json新增如下配置

{  "devDependencies": {    "@argo-design/prettier-config": "^1.0.0"  },  "prettier": "@argo-design/prettier-config"}

運行yarn重新安裝依賴。

5. VSCode安裝prettier擴展并將其設置成默認格式化工具

// settings.json{  "editor.defaultFormatter": "esbenp.prettier-vscode"}

集成stylelint

stylelint配置我們也新建一個package@argo-design/stylelint-config。

1. 安裝

cd argo-stylelint-configyarn add stylelint stylelint-prettier stylelint-config-prettier stylelint-order stylelint-config-rational-order postcss-html postcss-less# 單獨postcss8yarn add postcss@^8.0.0

對于結合prettier這里不在贅述。

stylelint-order允許我們自定義樣式屬性名稱順序。而stylelint-config-rational-order為我們提供了一套合理的開箱即用的順序。

值得注意的是,stylelint14版本不在默認支持less,sass等預處理語言。并且stylelint14依賴postcss8版本,可能需要單獨安裝,否則vscode 的stylellint擴展可能提示報錯TypeError: this.getPosition is not a function at LessParser.inlineComment....

2. 使用

// argo-stylelint-config/index.jsmodule.exports = {  plugins: [    "stylelint-prettier",  ],  extends: [    // "stylelint-config-standard",    "stylelint-config-standard-vue",     "stylelint-config-rational-order",    "stylelint-prettier/recommended"  ],  rules: {    "length-zero-no-unit": true, // 值為0不需要單位    "plugin/rational-order": [      true,      {        "border-in-box-model": true, // Border理應作為盒子模型的一部分 默認false        "empty-line-between-groups": false // 組之間添加空行 默認false      }    ]  },  overrides: [    {      files: ["*.html", "**/*.html"],      customSyntax: "postcss-html"    },    {      files: ["**/*.{less,css}"],      customSyntax: "postcss-less"    }  ]};

3. 集成到項目全局

根目錄package.json新增如下配置

{  "devDependencies": {    "@argo-design/stylelint-config": "^1.0.0"  },  "stylelint": {    "extends": [      "@argo-design/stylelint-config"    ]  }}

運行yarn重新安裝依賴。

4. VSCode保存時自動修復

VSCode安裝Stylelint擴展并添加配置

// settings.json{  "editor.codeActionsOnSave": {    "source.fixAll.eslint": true,    "source.fixAll.stylelint": true  },  "stylelint.validate": ["css", "less", "vue", "html"],  "css.validate": false,  "less.validate": false}

修改settings.json之后如不能及時生效,可以重啟一下vscode。如果你喜歡,可以將eslint,prettier,stylelint配置安裝到全局并集成到編輯器。

集成husky

為防止一些非法的commitpush,我們借助git hooks工具在對代碼提交前進行 ESLint 與 Stylelint的校驗,如果校驗通過,則成功commit,否則取消commit。

1. 安裝

# 在根目錄安裝huskyyarn add husky -D -W

2. 使用

npm pkg set scripts.prepare="husky install"npm run prepare# 添加pre-commit鉤子,在提交前運行代碼lintnpx husky add .husky/pre-commit "yarn lint"

至此,當我們執(zhí)行git commit -m "xxx"時就會先執(zhí)行l(wèi)int校驗我們的代碼,如果lint通過,成功commit,否則終止commit。具體的lint命令請自行添加。

集成lint-staged: 僅校驗staged中文件

現(xiàn)在,當我們git commit時,會對整個工作區(qū)的代碼進行l(wèi)int。當工作區(qū)文件過多,lint的速度就會變慢,進而影響開發(fā)體驗。實際上我們只需要對暫存區(qū)中的文件進行l(wèi)int即可。下面我們引入·lint-staged解決我們的問題。

1. 安裝

在根目錄安裝lint-staged

yarn add lint-staged -D -W

2. 使用

在根目錄package.json中添加如下的配置:

{  "lint-staged": {    "*.{js,ts,jsx,tsx}": [      "eslint --fix",      "prettier --write"    ],    "*.{less,css}": [      "stylelint --fix",      "prettier --write"    ],    "**/*.vue": [      "eslint --fix",      "stylelint --fix",      "prettier --write"    ]  }}

在monorepo中,lint-staged運行時,將始終向上查找并應用最接近暫存文件的配置,因此我們可以在根目錄下的package.json中配置lint-staged。值得注意的是,每個glob匹配的數(shù)組中的命令是從左至右依次運行,和webpack的loder應用機制不同!

最后,我們在.husky文件夾中找到pre-commit,并將yarn lint修改為npx --no-install lint-staged

#!/usr/bin/env sh. "$(dirname -- "$0")/_/husky.sh"npx --no-install lint-staged

至此,當我們執(zhí)行git commit -m "xxx"時,lint-staged會如期運行幫我們校驗staged(暫存區(qū))中的代碼,避免了對工作區(qū)的全量檢查。

集成commitlint: 規(guī)范化commit message

除了代碼規(guī)范檢查之后,Git 提交信息的規(guī)范也是不容忽視的一個環(huán)節(jié),規(guī)范精準的 commit 信息能夠方便自己和他人追蹤項目和把控進度。這里,我們使用大名鼎鼎的Angular團隊提交規(guī)范。

commit message格式規(guī)范

commit message 由 HeaderBodyFooter組成。其中Herder時必需的,Body和Footer可選。

Header

Header 部分包括三個字段 type、scopesubject。

<type>(<scope>): <subject>
type

其中type 用于說明 commit 的提交類型(必須是以下幾種之一)。

描述
featFeature) 新增一個功能
fixBug修復
docsDocumentation) 文檔相關
style代碼格式(不影響功能,例如空格、分號等格式修正),并非css樣式更改
refactor代碼重構
perfPerforment) 性能優(yōu)化
test測試相關
build構建相關(例如 scopes: webpack、gulp、npm 等)
ci更改持續(xù)集成軟件的配置文件和 package 中的 scripts 命令,例如 scopes: Travis, Circle 等
chore變更構建流程或輔助工具,日常事務
revertgit revert
scope

scope 用于指定本次 commit 影響的范圍。

subject

subject 是本次 commit 的簡潔描述,通常遵循以下幾個規(guī)范:

用動詞開頭,第一人稱現(xiàn)在時表述,例如:change 代替 changed 或 changes

第一個字母小寫

結尾不加句號.

Body(可選)

body 是對本次 commit 的詳細描述,可以分成多行。跟 subject 類似。

Footer(可選)

如果本次提交的代碼是突破性的變更或關閉Issue,則 Footer 必需,否則可以省略。

集成commitizen(可選)

我們可以借助工具幫我們生成規(guī)范的message。

1. 安裝

yarn add commitizen -D -W

2. 使用

安裝適配器

yarn add cz-conventional-changelog -D -W

這行命令做了兩件事:

安裝cz-conventional-changelog到開發(fā)依賴

在根目錄下的package.json中增加了:

"config": {  "commitizen": {    "path": "./node_modules/cz-conventional-changelog"  }}

添加npm scriptscm

"scripts": {  "cm": "cz"},

至此,執(zhí)行yarn cm,就能看到交互界面了!跟著交互一步步操作就能自動生成規(guī)范的message了。

集成commitlint: 對最終提交的message進行校驗

1. 安裝

首先在根目錄安裝依賴:

yarn add commitlint @commitlint/cli @commitlint/config-conventional -D -W

2. 使用

接著新建.commitlintrc.js:

module.exports = {  extends: ["@commitlint/config-conventional"]};

最后向husky中添加commit-msg鉤子,終端執(zhí)行:

npx husky add .husky/commit-msg "npx --no-install commitlint -e $HUSKY_GIT_PARAMS"

執(zhí)行成功之后就會在.husky文件夾中看到commit-msg文件了:

#!/usr/bin/env sh. "$(dirname -- "$0")/_/husky.sh"npx --no-install commitlint -e

至此,當你提交代碼時,如果pre-commit鉤子運行成功,緊接著在commit-msg鉤子中,commitlint會如期運行對我們提交的message進行校驗。

關于lint工具的集成到此就告一段落了,在實際開發(fā)中,我們還會對lint配置進行一些小改動,比如ignore,相關rules等等。這些和具體項目有關,我們不會變更package里的配置。

千萬別投機取巧拷貝別人的配置文件!復制一時爽,代碼火葬場。

圖標庫

巧婦難為無米之炊。組件庫通常依賴很多圖標,因此我們先開發(fā)一個支持按需引入的圖標庫。

假設我們現(xiàn)在已經拿到了一些漂亮的svg圖標,我們要做的就是將每一個圖標轉化生成.vue組件與一個組件入口index.ts文件。然后再生成匯總所有組件的入口文件。比如我們現(xiàn)在有foo.svg與bar.svg兩個圖標,最終生成的文件及結構如下:

相應的內容如下:

// bar.tsimport _Bar from "./bar.vue";const Bar = Object.assign(_Bar, {  install: (app) => {    app.component(_Bar.name, _Bar);  }});export default Bar;
// foo.tsimport _Foo from "./foo.vue";const Foo = Object.assign(_Foo, {  install: (app) => {    app.component(_Foo.name, _Foo);  }});export default Foo;
// argoIcon.tsimport Foo from "./foo";import Bar from "./bar";const icons = [Foo, Bar];const install = (app) => {  for (const key of Object.keys(icons)) {    app.use(icons[key]);  }};const ArgoIcon = {  ...icons,  install};export default ArgoIcon;
// index.tsexport { default } from "./argoIcon";export { default as Foo } from "./foo";export { default as Bar } from "./bar";

之所以這么設計是由圖標庫最終如何使用決定的,除此之外argoIcon.ts也將會是打包umd的入口文件。

// 全量引入import ArgoIcon from "圖標庫";app.use(ArgoIcon); // 按需引入import { Foo } from "圖標庫";app.use(Foo);

圖標庫的整個構建流程大概分為以下3步:

1. svg圖片轉.vue文件

整個流程很簡單,我們通過glob匹配到.svg拿到所有svg的路徑,對于每一個路徑,我們讀取svg的原始文本信息交由第三方庫svgo處理,期間包括刪除無用代碼,壓縮,自定義屬性等,其中最重要的是為svg標簽注入我們想要的自定義屬性,就像這樣:

<svg   :class="cls"   :style="innerStyle"  :stroke-linecap="strokeLinecap"  :stroke-linejoin="strokeLinejoin"  :stroke-width="strokeWidth">  <path d="..."></path></svg>

之后這段svgHtml會傳送給我們預先準備好的摸板字符串:

const template = `<template>  ${svgHtml}</template><script setup>defineProps({    "stroke-linecap": String;    // ...  })  // 省略邏輯代碼...</script>`

為摸板字符串填充數(shù)據(jù)后,通過fs模塊的writeFile生成我們想要的.vue文件。

2. 打包vue組件

在打包構建方案上直接選擇vite為我們提供的lib模式即可,開箱即用,插件擴展(后面會講到),基于rollup,能幫助我們打包生成ESM,這是按需引入的基礎。當然,commonjsumd也是少不了的。整個過程我們通過Vite 的JavaScript API實現(xiàn):

import { build } from "vite";import fs from "fs-extra";const CWD = process.cwd();const ES_DIR = resolve(CWD, "es");const LIB_DIR = resolve(CWD, "lib");interface compileOptions {  umd: boolean;  target: "component" | "icon";}async function compileComponent({  umd = false,  target = "component"}: compileOptions): Promise<void> {  await fs.emptyDir(ES_DIR);  await fs.emptyDir(LIB_DIR);  const config = getModuleConfig(target);  await build(config);  if (umd) {    await fs.emptyDir(DIST_DIR);    const umdConfig = getUmdConfig(target);    await build(umdConfig);  }}
import { InlineConfig } from "vite";import glob from "glob";const langFiles = glob.sync("components/locale/lang/*.ts");export default function getModuleConfig(type: "component" | "icon"): InlineConfig {  const entry = "components/index.ts";  const input = type === "component" ? [entry, ...langFiles] : entry;  return {    mode: "production",    build: {      emptyOutDir: true,      minify: false,      brotliSize: false,      rollupOptions: {        input,        output: [          {            format: "es", // 打包模式            dir: "es", // 產物存放路徑            entryFileNames: "[name].js", // 入口模塊的產物文件名            preserveModules: true, // 保留模塊結構,否則所有模塊都將打包在一個bundle文件中            /*             * 保留模塊的根路徑,該值會在打包后的output.dir中被移除             * 我們的入口是components/index.ts,打包后文件結構為:es/components/index.js             * preserveModulesRoot設為"components",打包后就是:es/index.js            */            preserveModulesRoot: "components"           },          {            format: "commonjs",            dir: "lib",            entryFileNames: "[name].js",            preserveModules: true,            preserveModulesRoot: "components",            exports: "named" // 導出模式          }        ]      },      // 開啟lib模式      lib: {        entry,        formats: ["es", "cjs"]      }    },    plugins: [      // 自定義external忽略node_modules      external(),      // 打包聲明文件      dts({        outputDir: "es",        entryRoot: C_DIR      })    ]  };};
export default function getUmdConfig(type: "component" | "icon"): InlineConfig {  const entry =    type === "component"      ? "components/argo-components.ts"      : "components/argo-icons.ts";  const entryFileName = type === "component" ? "argo" : "argo-icon";  const name = type === "component" ? "Argo" : "ArgoIcon";  return {    mode: "production",    build: {      target: "modules", // 支持原生 ES 模塊的瀏覽器      outDir: "dist", // 打包產物存放路徑      emptyOutDir: true, // 如果outDir在根目錄下,則清空outDir      sourcemap: true, // 生成sourcemap       minify: false, // 是否壓縮      brotliSize: false, // 禁用 brotli 壓縮大小報告。      rollupOptions: { // rollup打包選項        external: "vue", // 匹配到的模塊不會被打包到bundle        output: [          {            format: "umd", // umd格式            entryFileNames: `${entryFileName}.js`, // 即bundle名            globals: {              /*               * format為umd/iife時,標記外部依賴vue,打包后以Vue取代               * 未定義時打包結果如下               * var ArgoIcon = function(vue2) {}(vue);               * rollup自動猜測是vue,但實際是Vue.這會導致報錯               * 定義后               * var ArgoIcon = function(vue) {}(Vue);              */              vue: "Vue"            }          },          {            format: "umd",            entryFileNames: `${entryFileName}.min.js`,            globals: {              vue: "Vue"            },            plugins: [terser()] // terser壓縮          },        ]      },      // 開啟lib模式      lib: {        entry, // 打包入口        name // 全局變量名      }    },    plugins: [vue(), vueJsx()]  };};
export const CWD = process.cwd();export const C_DIR = resolve(CWD, "components");

可以看到,我們通過type區(qū)分組件庫和圖標庫打包。實際上打包圖標庫和組件庫都是差不多的,組件庫需要額外打包國際化相關的語言包文件。圖標樣式內置在組件之中,因此也不需要額外打包。

3. 打包聲明文件

我們直接通過第三方庫 vite-plugin-dts 打包圖標庫的聲明文件。

import dts from "vite-plugin-dts";plugins: [  dts({    outputDir: "es",    entryRoot: C_DIR  })]

關于打包原理可參考插件作者的這片文章。

lequ7.com/guan-yu-qia…

4. 實現(xiàn)按需引入

我們都知道實現(xiàn)tree-shaking的一種方式是基于ESM的靜態(tài)性,即在編譯的時候就能摸清依賴之間的關系,對于"孤兒"會殘忍的移除。但是對于import "icon.css"這種沒導入導出的模塊,打包工具并不知道它是否具有副作用,索性移除,這樣就導致頁面缺少樣式了。sideEffects就是npm與構建工具聯(lián)合推出的一個字段,旨在幫助構建工具更好的為npm包進行tree-shaking。

使用上,sideEffects設置為false表示所有模塊都沒有副作用,也可以設置數(shù)組,每一項可以是具體的模塊名或Glob匹配。因此,實現(xiàn)圖標庫的按需引入,只需要在argo-icons項目下的package.json里添加以下配置即可:

{  "sideEffects": false,}

這將告訴構建工具,圖標庫沒有任何副作用,一切沒有被引入的代碼或模塊都將被移除。前提是你使用的是ESM。

指定入口

Last but important!當圖標庫在被作為npm包導入時,我們需要在package.json為其配置相應的入口文件。

{  "main": "lib/index.js", // 以esm形式被引入時的入口  "module": "es/index.js", // 以commonjs形式被引入時的入口  "types": "es/index.d.ts" // 指定聲明文件}

引入storybook:是時候預覽我們的成果了!

顧名思義,storybook就是一本"書",講了很多個"故事"。在這里,"書"就是argo-icons,我為它講了3個故事:

基本使用

按需引入

使用iconfont.cn項目

初始化storybook

新建@argo-design/ui-storybookpackage,并在該目錄下運行:

npx storybook init -t vue3 -b webpack5

-t (即--type): 指定項目類型,storybook會根據(jù)項目依賴及配置文件等推算項目類型,但顯然我們僅僅是通過npm init新創(chuàng)建的項目,storybook無法自動判斷項目類型,故需要指定type為vue3,然后storybook會幫我們初始化storybook vue3 app。

-b (--builder): 指定構建工具,默認是webpack4,另外支持webpack5, vite。這里指定webpack5,否則后續(xù)會有類似報錯:cannot read property of undefine(reading "get")...因為storybook默認以webpack4構建,但是@storybook/vue3依賴webpack5,會沖突導致報錯。這里是天坑??!

storybook默認使用yarn安裝,如需指定npm請使用--use-npm。

這行命令主要幫我們做以下事情:

注入必要的依賴到packages.json(如若沒有指定-s,將幫我們自動安裝依賴)。

注入啟動,打包項目的腳本。

添加Storybook配置,詳見.storybook目錄。

添加Story范例文件以幫助我們上手,詳見stories目錄。

其中1,2步具體代碼如下:

{  "scripts": {    "storybook": "start-storybook -p 6006",    "build-storybook": "build-storybook"  },  "devDependencies": {    "@storybook/vue3": "^6.5.13",    "@storybook/addon-links": "^6.5.13",    "@storybook/addon-essentials": "^6.5.13",    "@storybook/addon-actions": "^6.5.13",    "@storybook/addon-interactions": "^6.5.13",    "@storybook/testing-library": "^0.0.13",    "vue-loader": "^16.8.3",    "@storybook/builder-webpack5": "^6.5.13",    "@storybook/manager-webpack5": "^6.5.13",    "@babel/core": "^7.19.6",    "babel-loader": "^8.2.5"  }}

接下來把目光放到.storybook下的main.js與preview.js

preview.js

preview.js可以具名導出parameters,decorators,argTypes,用于全局配置UI(stories,界面,控件等)的渲染行為。比如默認配置中的controls.matchers:

export const parameters = {  controls: {    matchers: {      color: /(background|color)$/i,      date: /Date$/    }  }};

它定義了如果屬性值是以background或color結尾,那么將為其啟用color控件,我們可以選擇或輸入顏色值,date同理。

除此之外你可以在這里引入全局樣式,注冊組件等等。更多詳情見官網 Configure story rendering

main.js

最后來看看最重要的項目配置文件。

module.exports = {  stories: [    "../stories/**/*.stories.mdx",    "../stories/**/*.stories.@(js|jsx|ts|tsx)"  ],  addons: [    "@storybook/addon-links",    "@storybook/addon-essentials",    "@storybook/addon-interactions"  ],  framework: "@storybook/vue3",  core: {    builder: "@storybook/builder-webpack5"  },}

stories, 即查找stroy文件的Glob。

addons, 配置需要的擴展。慶幸的是,當前一些重要的擴展都已經集成到@storybook/addon-essentials。

framework和core即是我們初識化傳遞的-t vue3 -b webpack5。

更多詳情見官網 Configure your Storybook project

配置并啟動storybook

less配置

由于項目使用到less因此我們需要配置一下less,安裝less以及相關loader。來到.storybook/main.js

module.exports = {  webpackFinal: (config) => {    config.module.rules.push({      test: /.less$/,      use: [        {          loader: "style-loader"        },        {          loader: "css-loader"        },        {          loader: "less-loader",          options: {            lessOptions: {              javascriptEnabled: true            }          }        }      ]    });    return config;  },}

配置JSX

storybook默認支持解析jsx/tsx,但你如果需要使用jsx書寫vue3的stories,仍需要安裝相關插件。

在argo-ui-storybook下安裝 @vue/babel-plugin-jsx

yarn add @vue/babel-plugin-jsx -D

新建.babelrc

{  "plugins": ["@vue/babel-plugin-jsx"]}

關于如何書寫story,篇幅受限,請自行查閱范例文件或官網。

配置完后終端執(zhí)行yarn storybook即可啟動我們的項目,辛苦的成果也將躍然紙上。

對于UI,在我們的組件庫逐漸豐富之后,將會自建一個獨具組件庫風格的文檔站點,拭目以待。

組件庫

組件通信

在Vue2時代,組件跨層級通信方式可謂“百花齊放”,provide/inject就是其中一種。時至今日,在composition,es6,ts加持下,provide/inject可以更加大展身手。

provide/inject原理

在創(chuàng)建組件實例時,會在自身掛載一個provides對象,默認指向父實例的provides。

const instance = {  provides: parent ? parent.provides : Object.create(appContext.provides)}

appContext.provides即createApp創(chuàng)建的app的provides屬性,默認是null

在自身需要為子組件供數(shù)據(jù)時,即調用provide()時,會創(chuàng)建一個新對象,該對象的原型指向父實例的provides,同時將provide提供的選項添加到新對象上,這個新對象就是實例新的provides值。代碼簡化就是

function provide(key, value) {   const parentProvides = currentInstance.parent && currentInstance.parent.provides;   const newObj = Object.create(parentProvides);  currentInstance.provides = newObj;  newObj[key] = value;}

而inject的實現(xiàn)原理則時通過key去查找祖先provides對應的值:

function inject(key, defaultValue) {   const instance = currentInstance;   const provides = instance.parent == null    ? instance.vnode.appContent && instance.vnode.appContent.provides    :instance.parent.provides;  if(provides && key in provides) {    return provides[key]  }}

你可能會疑惑,為什么這里是直接去查父組件,而不是先查自身實例的provides呢?前面不是說實例的provides默認指向父實例的provides么。但是請注意,是“默認”。如果當前實例執(zhí)行了provide()是不是把instance.provides“污染”了呢?這時再執(zhí)行inject(key),如果provide(key)的key與你inject的key一致,就從當前實例provides取key對應的值了,而不是取父實例的provides!

最后,我畫了2張圖幫助大家理解

新增button組件并完成打包

篇幅有限,本文不會對組件的具體實現(xiàn)講解哦,簡單介紹下文件

__demo__組件使用事例constants.ts定義的常量context.ts上下文相關interface.ts組件接口TEMPLATE.md用于生成README.md的模版button/style下存放組件樣式style下存放全局樣式

打包esm與commonjs模塊

關于打包組件的esmcommonjs模塊在之前打包圖標庫章節(jié)已經做了介紹,這里不再贅述。

打包樣式

相對于圖標庫,組件庫的打包需要額外打包樣式文件,大概流程如下:

生成總入口components/index.less并編譯成css。

編譯組件less。

生成dist下的argo.css與argo.min.css。

構建組件style/index.ts。

1. 生成總入口components/index.less

import path from "path";import { outputFileSync } from "fs-extra";import glob from "glob";export const CWD = process.cwd();export const C_DIR = path.resolve(CWD, "components");export const lessgen = async () => {  let lessContent = `@import "./style/index.less";\n`; // 全局樣式文件  const lessFiles = glob.sync("**/style/index.less", {    cwd: C_DIR,    ignore: ["style/index.less"]  });  lessFiles.forEach((value) => {    lessContent += `@import "./${value}";\n`;  });  outputFileSync(path.resolve(C_DIR, "index.less"), lessContent);  log.success("genless", "generate index.less success!");};

代碼很簡單,值得一提就是為什么不將lessContent初始化為空,glob中將ignore移除,這不是更簡潔嗎。這是因為style/index.less作為全局樣式,我希望它在引用的最頂部。最終將會在components目錄下生成index.less內容如下:

@import "./style/index.less";@import "./button/style/index.less";/* other less of components */

2. 打包組件樣式

import path from "path";import { readFile, copySync } from "fs-extra"import { render } from "less";export const ES_DIR = path.resolve(CWD, "es");export const LIB_DIR = path.resolve(CWD, "lib");const less2css = (lessPath: string): string => {  const source = await readFile(lessPath, "utf-8");  const { css } = await render(source, { filename: lessPath });  return css;}const files = glob.sync("**/*.{less,js}", {  cwd: C_DIR});for (const filename of files) {  const lessPath = path.resolve(C_DIR, `${filename}`);  // less文件拷貝到es和lib相對應目錄下  copySync(lessPath, path.resolve(ES_DIR, `${filename}`));  copySync(lessPath, path.resolve(LIB_DIR, `${filename}`));  // 組件樣式/總入口文件/全局樣式的入口文件編譯成css  if (/index.less$/.test(filename)) {    const cssFilename = filename.replace(".less", ".css");    const ES_DEST = path.resolve(ES_DIR, `${cssFilename}`);    const LIB_DEST = path.resolve(LIB_DIR, `${cssFilename}`);    const css = await less2css(lessPath);    writeFileSync(ES_DEST, css, "utf-8");    writeFileSync(LIB_DEST, css, "utf-8");  }}

3. 生成dist下的argo.css與argo.min.css

import path from "path";import CleanCSS, { Output } from "clean-css";import { ensureDirSync } from "fs-extra";export const DIST_DIR = path.resolve(CWD, "dist");console.log("start build components/index.less to dist/argo(.min).css");const indexCssPath = path.resolve(ES_DIR, "index.css");const css = readFileSync(indexCssPath, "utf8");const minContent: Output = new CleanCSS().minify(css);ensureDirSync(DIST_DIR);writeFileSync(path.resolve("dist/argo.css"), css);writeFileSync(path.resolve("dist/argo.min.css"), minContent.styles);log.success(`build components/index.less to dist/argo(.min).css`);

其中最重要的就是使用clean-css壓縮css。

4. 構建組件style/index.ts

如果你使用過babel-plugin-import,那一定熟悉這項配置:

["import", { "libraryName": "antd", "style": true }]: import js and css modularly (LESS/Sass source files)["import", { "libraryName": "antd", "style": "css" }]: import js and css modularly (css built files)

通過指定style: true,babel-plugin-import可以幫助我們自動引入組件的less文件,如果你擔心less文件定義的變量會被覆蓋或沖突,可以指定"css",即可引入組件的css文件樣式。

這一步就是要接入這點。但目前不是很必要,且涉及到vite插件開發(fā),暫可略過,后面會講。

來看看最終實現(xiàn)的樣子。

其中button/style/index.js內容也就是導入less:

import "../../style/index.less";import "./index.less";

button/style/css.js內容也就是導入css:

import "../../style/index.css";import "./index.css";

最后你可能會好奇,諸如上面提及的compileComponent,compileStyle等函數(shù)是如何被調度使用的,這其實都歸功于腳手架@argo-design/scripts。當它作為依賴被安裝到項目中時,會為我們提供諸多命令如argo-scripts geniconargo-scripts compileComponent等,這些函數(shù)都在執(zhí)行命令時被調用。

配置sideEffects

"sideEffects": [  "dist/*",  "es/**/style/*",  "lib/**/style/*",  "*.less"]

國際化

基本實現(xiàn)

// locale.tsimport { ref, reactive, computed, inject } from "vue";import { isString } from "../_utils/is";import zhCN from "./lang/zh-cn";export interface ArgoLang {  locale: string;  button: {    defaultText: string;  }}type ArgoI18nMessages = Record<string, ArgoLang>;// 默認使用中文const LOCALE = ref("zh-CN");const I18N_MESSAGES = reactive<ArgoI18nMessages>({  "zh-CN": zhCN});// 添加語言包export const addI18nMessages = (  messages: ArgoI18nMessages,  options?: {    overwrite?: boolean;  }) => {  for (const key of Object.keys(messages)) {    if (!I18N_MESSAGES[key] || options?.overwrite) {      I18N_MESSAGES[key] = messages[key];    }  }};// 切換語言包export const useLocale = (locale: string) => {  if (!I18N_MESSAGES[locale]) {    console.warn(`use ${locale} failed! Please add ${locale} first`);    return;  }  LOCALE.value = locale;};// 獲取當前語言export const getLocale = () => {  return LOCALE.value;};export const useI18n = () => {  const i18nMessage = computed<ArgoLang>(() => I18N_MESSAGES[LOCALE.value]);  const locale = computed(() => i18nMessage.value.locale);  const transform = (key: string): string => {    const keyArray = key.split(".");    let temp: any = i18nMessage.value;    for (const keyItem of keyArray) {      if (!temp[keyItem]) {        return key;      }      temp = temp[keyItem];    }    return temp;  };  return {    locale,    t: transform  };};

添加需要支持的語言包,這里默認支持中文和英文。

// lang/zh-CN.tsconst lang: ArgoLang = {  locale: "zh-CN",  button: {    defaultText: "按鈕"  },}
// lang/en-US.tsconst lang: ArgoLang = {  locale: "en-US",  button: {    defaultText: "Button",  },}

button組件中接入

<template>  <button>    <slot> {{ t("button.defaultText") }} </slot>  </button></template><script>import { defineComponent } from "vue";import { useI18n } from "../locale";export default defineComponent({  name: "Button",  setup(props, { emit }) {    const { t } = useI18n();    return {      t    };  }});</script>

Button的國際化僅做演示,實際上國際化在日期日歷等組件中才有用武之地。

國際化演示

argo-ui-storybook/stories中添加locale.stories.ts

import { computed } from "vue";import { Meta, StoryFn } from "@storybook/vue3";import {  Button,  addI18nMessages,  useLocale,  getLocale} from "@argo-design/argo-ui/components/index"; // 源文件形式引入方便開發(fā)時調試import enUS from "@argo-design/argo-ui/components/locale/lang/en-us";interface Args {}export default {  title: "Component/locale",  argTypes: {}} as Meta<Args>;const BasicTemplate: StoryFn<Args> = (args) => {  return {    components: { Button },    setup() {      addI18nMessages({ "en-US": enUS });      const currentLang = computed(() => getLocale());      const changeLang = () => {        const lang = getLocale();        if (lang === "en-US") {          useLocale("zh-CN");        } else {          useLocale("en-US");        }      };      return { args, changeLang, currentLang };    },    template: `      <h1>內部切換語言,當前語言: {{currentLang}}</h1>      <p>僅在未提供ConfigProvider時生效</p>      <Button type="primary" @click="changeLang">點擊切換語言</Button>      <Button long style="marginTop: 20px;"></Button>    `  };};export const Basic = BasicTemplate.bind({});Basic.storyName = "基本使用";Basic.args = {};

.preview.js中全局引入組件庫樣式

import "@argo-design/argo-ui/components/index.less";

終端啟動項目就可以看到效果了。

實現(xiàn)config-provider組件

通常組件庫都會提供config-provider組件來使用國際化,就像下面這樣

<template>  <a-config-provider :locale="enUS">    <a-button />  </a-config-provider></template>

下面我們來實現(xiàn)一下config-provider組件:

<template>  <slot /></template><script>import type { PropType } from "vue";import {  defineComponent,  provide,  reactive,  toRefs,} from "vue";import { configProviderInjectionKey } from "./context";export default defineComponent({  name: "ConfigProvider",  props: {    locale: {      type: Object as PropType<ArgoLang>    },  },  setup(props, { slots }) {    const { locale } = toRefs(props);    const config = reactive({      locale,    });    provide(configProviderInjectionKey, config);  }});</script>
export interface ConfigProvider {  locale?: ArgoLang;}export const configProviderInjectionKey: InjectionKey<ConfigProvider> =  Symbol("ArgoConfigProvider");

修改locale/index.ts中計算屬性i18nMessage的獲取邏輯

import { configProviderInjectionKey } from "../config-provider/context";export const useI18n = () => {  const configProvider = inject(configProviderInjectionKey, undefined);  const i18nMessage = computed<ArgoLang>(    () => configProvider?.locale ?? I18N_MESSAGES[LOCALE.value]  );  const locale = computed(() => i18nMessage.value.locale);  const transform = (key: string): string => {    const keyArray = key.split(".");    let temp: any = i18nMessage.value;    for (const keyItem of keyArray) {      if (!temp[keyItem]) {        return key;      }      temp = temp[keyItem];    }    return temp;  };  return {    locale,    t: transform  };};

編寫stories驗證一下:

const ProviderTemplate: StoryFn<Args> = (args) => {  return {    components: { Button, ConfigProvider },    render() {      return (        <ConfigProvider {...args}>          <Button long={true} />        </ConfigProvider>      );    }  };};export const Provider = ProviderTemplate.bind({});Provider.storyName = "在config-provider中使用";Provider.args = {  // 在這里把enUS傳給ConfigProvider的locale  locale: enUS};

以上stories使用到了jsx,請確保安裝并配置了@vue/babel-plugin-jsx

可以看到,Button默認是英文的,表單控件也接收到enUS語言包了,符合預期。

自動引入組件樣式

值得注意的是,上面提到的按需引入只是引入了組件js邏輯代碼,但對于樣式依然沒有引入。

下面我們通過開發(fā)vite插件vite-plugin-auto-import-style,讓組件庫可以自動引入組件樣式。

效果演示

現(xiàn)在我們書寫的代碼如下,現(xiàn)在我們已經知道了,這樣僅僅是加載了組件而已。

import { createApp } from "vue";import App from "./App.vue";import { Button, Empty, ConfigProvider } from "@argo-design/argo-ui";import { Anchor } from "@argo-design/argo-ui";createApp(App)  .use(Button)  .use(Empty)  .use(ConfigProvider)  .use(Anchor)  .mount("#root");

添加插件之前:

添加插件之后:

import { defineConfig } from "vite";import argoAutoInjectStyle from "vite-plugin-argo-auto-inject-style";export default defineConfig({  plugins: [      argoAutoInjectStyle({            libs: [        {                  libraryName: "@argo-design/argo-ui",                            resolveStyle: (name) => {                              return `@argo-design/argo-ui/es/${name}/style/index.js`;          }        }      ]    })  ]})

插件實現(xiàn)

實踐之前瀏覽一遍官網插件介紹是個不錯的選擇。插件API

vite插件是一個對象,通常由name和一系列鉤子函數(shù)組成:

{  name: "vite-plugin-vue-auto-inject-style",  configResolved(config) {}}

常用鉤子

config

vite.config.ts被解析完成后觸發(fā)。常用于擴展配置??梢灾苯釉赾onfig上定義或返回一個對象,該對象會嘗試與配置文件vite.config.ts中導出的配置對象深度合并。

configResolved

在解析完所有配置時觸發(fā)。形參config表示最終確定的配置對象。通常將該配置保存起來在有需要時提供給其它鉤子使用。

resolveId

開發(fā)階段每個傳入模塊請求時被調用,常用于解析模塊路徑。返回string或對象將終止后續(xù)插件的resolveId鉤子執(zhí)行。

load

resolveId之后調用,可自定義模塊加載內容

transform

load之后調用,可自定義修改模塊內容。這是一個串行鉤子,即多個插件實現(xiàn)了這個鉤子,下個插件的transform需要等待上個插件的transform鉤子執(zhí)行完畢。上個transform返回的內容將傳給下個transform鉤子。

為了讓插件完成自動引入組件樣式,我們需要完成如下工作:

過濾出我們想要的文件。

對文件內容進行AST解析,將符合條件的import語句提取出來。

然后解析出具體import的組件。

最后根據(jù)組件查找到樣式文件路徑,生成導入樣式的語句字符串追加到import語句后面即可。

其中過濾我們使用rollup提供的工具函數(shù)createFilter;

AST解析借助es-module-lexer,非常出名,千萬級周下載量。

import type { Plugin } from "vite";import { createFilter } from "@rollup/pluginutils";import { ExportSpecifier, ImportSpecifier, init, parse } from "es-module-lexer";import MagicString from "magic-string";import * as changeCase from "change-case";import { Lib, VitePluginOptions } from "./types";const asRE = /\s+as\s+\w+,?/g;// 插件本質是一個對象,但為了接受在配置時傳遞的參數(shù),我們通常在一個函數(shù)中將其返回。// 插件默認開發(fā)和構建階段都會應用export default function(options: VitePluginOptions): Plugin {  const {    libs,    include = ["**/*.vue", "**/*.ts", "**/*.tsx"],    exclude = "node_modules/**"  } = options;  const filter = createFilter(include, exclude);  return {    name: "vite:argo-auto-inject-style",    async transform(code: string, id: string) {      if (!filter(id) || !code || !needTransform(code, libs)) {        return null;      }      await init;      let imports: readonly ImportSpecifier[] = [];      imports = parse(code)[0];        if (!imports.length) {        return null;      }      let s: MagicString | undefined;      const str = () => s || (s = new MagicString(code));      for (let index = 0; index < imports.length; index++) {        // ss import語句開始索引        // se import語句介結束索引        const { n: moduleName, se, ss } = imports[index];        if (!moduleName) continue;        const lib = getLib(moduleName, libs);        if (!lib) continue;        // 整條import語句        const importStr = code.slice(ss, se);         // 拿到每條import語句導入的組件集合        const importItems = getImportItems(importStr);        let endIndex = se + 1;        for (const item of importItems) {          const componentName = item.n;          const paramName = changeCase.paramCase(componentName);          const cssImportStr = `\nimport "${lib.resolveStyle(paramName)}";`;          str().appendRight(endIndex, cssImportStr);        }      }      return {        code: str().toString()      };    }  };}export type { Lib, VitePluginOptions };function getLib(libraryName: string, libs: Lib[]) {  return libs.find((item) => item.libraryName === libraryName);}function getImportItems(importStr: string) {  if (!importStr) {    return [];  }  const matchItem = importStr.match(/{(.+?)}/gs);  const formItem = importStr.match(/from.+/gs);  if (!matchItem) return [];  const exportStr = `export ${matchItem[0].replace(asRE, ",")} ${formItem}`;  let importItems: readonly ExportSpecifier[] = [];  try {    importItems = parse(exportStr)[1];  } catch (error) {    console.log(error);  }  return importItems;}function needTransform(code: string, libs: Lib[]) {  return libs.some(({ libraryName }) => {    return new RegExp(`("${libraryName}")|("${libraryName}")`).test(code);  });}
export interface Lib {  libraryName: string;  resolveStyle: (name: string) => string;}export type RegOptions =  | string  | RegExp  | Array<string | RegExp>  | null  | undefined;export interface VitePluginOptions {  include?: RegOptions;  exclude?: RegOptions;  libs: Lib[];}

換膚與暗黑風格

換膚

在我們的less樣式中,會定義一系列如下的顏色梯度變量,其值由color-palette函數(shù)完成:

@blue-6: #3491fa;@blue-1: color-palette(@blue-6, 1);@blue-2: color-palette(@blue-6, 2);@blue-3: color-palette(@blue-6, 3);@blue-4: color-palette(@blue-6, 4);@blue-5: color-palette(@blue-6, 5);@blue-7: color-palette(@blue-6, 7);@blue-8: color-palette(@blue-6, 8);@blue-9: color-palette(@blue-6, 9);@blue-10: color-palette(@blue-6, 10);

基于此,我們再演化出具體場景下的顏色梯度變量:

@primary-1: @blue-1;@primary-2: @blue-2;@primary-3: @blue-3;// 以此類推...@success-1: @green-1;@success-2: @green-2;@success-3: @green-3;// 以此類推.../* @warn @danger @info等等 */

有了具體場景下的顏色梯度變量,我們就可以設計變量供給組件消費了:

@color-primary-1: @primary-1;@color-primary-2: @primary-2;@color-primary-3: @primary-3;/* ... */
.argo-btn.arco-btn-primary {  color: #fff;    background-color: @color-primary-1;}

在使用組件庫的項目中我們通過 Less 的 ·modifyVars功能修改變量值:

Webpack配置

// webpack.config.jsmodule.exports = {  rules: [{    test: /.less$/,    use: [{      loader: "style-loader",    }, {      loader: "css-loader",    }, {      loader: "less-loader",     options: {       lessOptions: {         modifyVars: {           "primary-6": "#f85959",         },         javascriptEnabled: true,       },     },    }],  }],}

vite配置

// vite.config.jsexport default {  css: {   preprocessorOptions: {     less: {       modifyVars: {         "primary-6": "#f85959",       },       javascriptEnabled: true,     }   }  },}

設計暗黑風格

首先,顏色梯度變量需要增加暗黑風格。也是基于@blue-6計算,只不過這里換成了dark-color-palette函數(shù):

@dark-blue-1: dark-color-palette(@blue-6, 1);@dark-blue-2: dark-color-palette(@blue-6, 2);@dark-blue-3: dark-color-palette(@blue-6, 3);@dark-blue-4: dark-color-palette(@blue-6, 4);@dark-blue-5: dark-color-palette(@blue-6, 5);@dark-blue-6: dark-color-palette(@blue-6, 6);@dark-blue-7: dark-color-palette(@blue-6, 7);@dark-blue-8: dark-color-palette(@blue-6, 8);@dark-blue-9: dark-color-palette(@blue-6, 9);@dark-blue-10: dark-color-palette(@blue-6, 10);

然后,在相應節(jié)點下掛載css變量

body {  --color-bg: #fff;    --color-text: #000;    --primary-6: @primary-6; }body[argo-theme="dark"] {  --color-bg: #000;    --color-text: #fff;    --primary-6: @dark-primary-6; }

緊接著,組件消費的less變量更改為css變量:

.argo-btn.argo-btn-primary {  color: #fff;    background-color: var(--primary-6);}

此外,我們還設置了--color-bg,--color-text等用于設置body色調:

body {  color: var(--color-bg);    background-color: var(--color-text);}

最后,在消費組件庫的項目中,通過編輯body的argo-theme屬性即可切換亮暗模式:

// 設置為暗黑模式document.body.setAttribute("argo-theme", "dark")// 恢復亮色模式document.body.removeAttribute("argo-theme");

在線動態(tài)換膚

前面介紹的是在項目打包時通過less配置修改less變量值達到換膚效果,有了css變量,我們可以實現(xiàn)在線動態(tài)換膚。默認的,打包過后樣式如下:

body {  --primary-6: "#3491fa"}.argo-btn {    color: #fff;    background-color: var(--primary-6);}

在用戶選擇相應顏色后,我們只需要更改css變量--primary-6的值即可:

// 可計算selectedColor的10個顏色梯度值列表,并逐一替換document.body.style.setProperty("--primary-6", colorPalette(selectedColor, 6));// ....

文檔站點

還記得每個組件目錄下的TEMPLATE.md文件嗎?

## zh-CN```yamlmeta:  type: 組件  category: 通用title: 按鈕 Buttondescription: 按鈕是一種命令組件,可發(fā)起一個即時操作。```---## en-US```yamlmeta:  type: Component  category: Commontitle: Buttondescription: Button is a command component that can initiate an instant operation.```---@import ./__demo__/basic.md@import ./__demo__/disabled.md## API%%API(button.vue)%%## TS%%TS(interface.ts)%%

它是如何一步步被渲染出我們想要的界面呢?

TEMPLATE.md的作用

TEMPLATE.md將被解析并生成中英文版READE.md(組件使用文檔),之后在vue-router中被加載使用。

這時當我們訪問路由/button,vite服務器將接管并調用一系列插件解析成瀏覽器識別的代碼,最后由瀏覽器渲染出我們的文檔界面。

1. 解析TEMPLATE 生成 README

簡單起見,我們忽略國際化和使用例子部分。

%%API(button.vue)%%%%INTERFACE(interface.ts)%%

其中button.vue就是我們的組件,interface.ts就是定義組件的一些接口,比如ButtonProps,ButtonType等。

解析button.vue

大致流程如下:

讀取TEMPLATE.md,正則匹配出button.vue;

使用vue-doc-api解析vue文件; let componentDocJson = VueDocApi.parse(path.resolve(__dirname, "button.vue"));

componentDocJson轉換成md字符串,md字符串替換掉占位符%%API(button.vue)%%,寫入README.md;

關于vue文件與解析出來的conponentDocJson結構見 vue-docgen-api

解析interface.ts

由于VueDocApi.parse無法直接解析.ts文件,因此借助ts-morph解析ts文件并轉換成componentDocJson結構的JSON對象,再將componentDocJson轉換成md字符串,替換掉占位符后最終寫入README.md;

讀取TEMPLATE.md,正則匹配出interface.ts;

使用ts-morph解析inerface.ts出interfaces;

interfaces轉componentDocJson;

componentDocJson轉換成md字符串,md字符串替換掉占位符%%API(button.vue)%%,寫入README.md;

import { Project } from "ts-morph";const project = new Project();project.addSourceFileAtPath(filepath);const sourceFile = project.getSourceFile(filepath);const interfaces = sourceFile.getInterfaces();const componentDocList = [];interfaces.forEach((interfaceDeclaration) => {  const properties = interfaceDeclaration.getProperties();  const componentDocJson = {    displayName: interfaceDeclaration.getName(),    exportName: interfaceDeclaration.getName(),    props: formatterProps(properties),    tags: {}  };  if (componentDocJson.props.length) {    componentDocList.push(componentDocJson);  }});// genMd(componentDocList);

最終生成README.zh-CN.md如下

```yamlmeta:  type: 組件  category: 通用title: 按鈕 Buttondescription: 按鈕是一種命令組件,可發(fā)起一個即時操作。```@import ./__demo__/basic.md@import ./__demo__/disabled.md## API### `<button>` Props|參數(shù)名|描述|類型|默認值||---|---|---|:---:||type|按鈕的類型,分為五種:次要按鈕、主要按鈕、虛框按鈕、線性按鈕、文字按鈕。|`"secondary" | "primary" | "dashed" | "outline" | "text"`|`"secondary"`||shape|按鈕的形狀|`"square" | "round" | "circle"`|`"square"`||status|按鈕的狀態(tài)|`"normal" | "warning" | "success" | "danger"`|`"normal"`||size|按鈕的尺寸|`"mini" | "small" | "medium" | "large"`|`"medium"`||long|按鈕的寬度是否隨容器自適應。|`boolean`|`false`||loading|按鈕是否為加載中狀態(tài)|`boolean`|`false`||disabled|按鈕是否禁用|`boolean`|`false`||html-type|設置 `button` 的原生 `type` 屬性,可選值參考 [HTML標準](https://developer.mozilla.org/en-US/docs/Web/HTML/Element/button#attr-type "_blank")|`"button" | "submit" | "reset"`|`"button"`||href|設置跳轉鏈接。設置此屬性時,按鈕渲染為a標簽。|`string`|`-`|### `<button>` Events|事件名|描述|參數(shù)||---|---|---||click|點擊按鈕時觸發(fā)|event: `Event`|### `<button>` Slots|插槽名|描述|參數(shù)||---|:---:|---||icon|圖標|-|### `<button-group>` Props|參數(shù)名|描述|類型|默認值||---|---|---|:---:||disabled|是否禁用|`boolean`|`false`|## INTERFACE### ButtonProps|參數(shù)名|描述|類型|默認值||---|---|---|:---:||type|按鈕類型|`ButtonTypes`|`-`|

2. 路由配置

const Button = () => import("@argo-design/argo-ui/components/button/README.zh-CN.md");const router = createRouter({  {    path: "/button",  component: Button  }});export default router;

3. README是如何被渲染成UI的

首先我們來看下README.md(為方便直接省略.zh-CN)以及其中的demos.md的樣子與它們最終的UI。

可以看到,README就是一系列demo的集合,而每個demo都會被渲染成一個由代碼示例與代碼示例運行結果組成的代碼塊。

開發(fā)vite-plugin-vue-docs解析md

yarn create vite快速搭建一個package

// vite.config.tsimport { defineConfig } from "vite";import vue from "@vitejs/plugin-vue";import md from "./plugins/vite-plugin-md/index";export default defineConfig({  server: {    port: 8002,  },  plugins: [md(), vue()],});
// App.vue<template>  <ReadMe /></template><script setup>import ReadMe from "./readme.md";</script>
// readme.md@import ./__demo__/basic.md

開發(fā)之前我們先看看插件對README.md源碼的解析轉換流程。

1. 源碼轉換

首先我們來實現(xiàn)第一步: 源碼轉換。即將

@import "./__demo__/basic.md"

轉換成

<template>  <basic-demo /></template><script>import { defineComponent } from "vue";import BasicDemo from "./__demo__/basic.md";export default defineComponent({  name: "ArgoMain",  components: { BasicDemo },});</script>

轉換過程我們借助第三方markdown解析工具marked完成,一個高速,輕量,無阻塞,多平臺的markdown解析器。

眾所周知,md2html規(guī)范中,文本默認會被解析渲染成p標簽。也就是說,README.md里的@import ./__demo__/basic.md會被解析渲染成<p>@import ./__demo__/basic.md</p>,這不是我想要的。所以需要對marked進行一下小小的擴展。

// marked.tsimport { marked } from "marked";import path from "path";const mdImport = {  name: "mdImport",  level: "block",  tokenizer(src: string) {    const rule = /^@import\s+(.+)(?:\n|$)/;    const match = rule.exec(src);    if (match) {      const filename = match[1].trim();      const basename = path.basename(filename, ".md");      return {        type: "mdImport",        raw: match[0],        filename,        basename,      };    }    return undefined;  },  renderer(token: any) {    return `<demo-${token.basename} />\n`;  },};marked.use({  extensions: [mdImport],});export default marked;

我們新建了一個mdImport的擴展,用來自定義解析我們的md。在tokenizer 中我們定義了解析規(guī)則并返回一系列自定義的tokens,其中raw就是@import "./__demo__/basic.md",filename就是./__demo__/basic.md,basename就是basic,我們可以通過marked.lexer(code)拿到這些tokens。在renderer中我們自定義了渲染的html,通過marked.parser(tokens)可以拿到html字符串了。因此,我們開始在插件中完成第一步。

// index.tsimport { Plugin } from "vite";import marked from "./marked";export default function vueMdPlugin(): Plugin {  return {    name: "vite:argo-vue-docs",    async transform(code: string, id: string) {      if (!id.endsWith(".md")) {        return null;      }      const tokens = marked.lexer(code);      const html = marked.parser(tokens);      const vueCode = transformMain({ html, tokens });    },  };}
// vue-template.tsimport changeCase from "change-case";import marked from "./marked";export const transformMain = ({  html,  tokens,}: {  html: string;  tokens: any[];}): string => {  const imports = [];  const components = [];  for (const token of tokens) {    const componentName = changeCase.pascalCase(`demo-${token.basename}`);    imports.push(`import ${componentName} from "${token.filename}";`);    components.push(componentName);  }  return `  <template>    ${html}  </template>  <script>import { defineComponent } from "vue";${imports.join("\n")};export default defineComponent({  name: "ArgoMain",  components: { ${components.join(",")} },});</script>`;};

其中change-case是一個名稱格式轉換的工具,比如basic-demo轉BasicDemo等。

transformMain返回的vueCode就是我們的目標vue模版了。但瀏覽器可不認識vue模版語法,所以我們仍要將其交給官方插件@vitejs/plugin-vuetransform鉤子函數(shù)轉換一下。

import { getVueId } from "./utils";export default function vueMdPlugin(): Plugin {  let vuePlugin: Plugin | undefined;  return {    name: "vite:argo-vue-docs",    configResolved(resolvedConfig) {      vuePlugin = resolvedConfig.plugins.find((p) => p.name === "vite:vue");    },    async transform(code: string, id: string) {      if (!id.endsWith(".md")) {        return null;      }      if (!vuePlugin) {        return this.error("Not found plugin [vite:vue]");      }      const tokens = marked.lexer(code);      const html = marked.parser(tokens);      const vueCode = transformMain({ html, tokens });      return await vuePlugin.transform?.call(this, vueCode, getVueId(id));    },  };}
// utils.tsexport const getVueId = (id: string) => {  return id.replace(".md", ".vue");};

這里使用getVueId修改擴展名為.vue是因為vuePlugin.transform會對非vue文件進行攔截就像我們上面攔截非md文件一樣。

configResolved鉤子函數(shù)中,形參resolvedConfig是vite最終使用的配置對象。在該鉤子中拿到其它插件并將其提供給其它鉤子使用,是vite插件開發(fā)中的一種“慣用伎倆”了。

2. 處理basic.md

在經過vuePlugin.transform及后續(xù)處理過后,最終vite服務器對readme.md響應給瀏覽器的內容如下

對于basic.md?import響應如下

可以看到,這一坨字符串可沒有有效的默認導出語句。因此對于解析語句import DemoBasic from "/src/__demo__/basic.md?import";瀏覽器會報錯

Uncaught SyntaxError: The requested module "/src/__demo__/basic.md?import" does not provide an export named "default" (at readme.vue:9:8)

在帶有module屬性的script標簽中,每個import語句都會向vite服務器發(fā)起請求進而繼續(xù)走到插件的transform鉤子之中。下面我們繼續(xù),對/src/__demo__/basic.md?import進行攔截處理。

// index.tsasync transform(code: string, id: string) {  if (!id.endsWith(".md")) {    return null;  }  // 新增對demo文檔的解析分支  if (isDemoMarkdown(id)) {    const tokens = marked.lexer(code);    const vueCode = transformDemo({ tokens, filename: id });    return await vuePlugin.transform?.call(this, vueCode, getVueId(id));  } else {    const tokens = marked.lexer(code);    const html = marked.parser(tokens);    const vueCode = transformMain({ html, tokens });    return await vuePlugin.transform?.call(this, vueCode, getVueId(id));  }},
// utils.tsexport const isDemoMarkdown = (id: string) => {  return //__demo__//.test(id);};
// vue-template.tsexport const transformDemo = ({  tokens,  filename,}: {  tokens: any[];  filename: string;}) => {  const data = {    html: "",  };  const vueCodeTokens = tokens.filter(token => {    return token.type === "code" && token.lang === "vue"  });  data.html = marked.parser(vueCodeTokens);  return `  <template>    <hr />    ${data.html}  </template>  <script>import { defineComponent } from "vue";export default defineComponent({  name: "ArgoDemo",});</script>`;};

現(xiàn)在已經可以在瀏覽器中看到結果了,水平線和示例代碼。

3. 虛擬模塊

那如何實現(xiàn)示例代碼的運行結果呢?其實在對tokens遍歷(filter)的時候,我們是可以拿到vue模版字符串的,我們可以將其緩存起來,同時手動構造一個import請求import Result from "${virtualPath}";這個請求用于返回運行結果。

export const transformDemo = ({  tokens,  filename,}: {  tokens: any[];  filename: string;}) => {  const data = {    html: "",  };  const virtualPath = `/@virtual${filename}`;  const vueCodeTokens = tokens.filter(token => {    const isValid = token.type === "code" && token.lang === "vue"    // 緩存vue模版代碼    isValid && createDescriptor(virtualPath, token.text);    return isValid;  });  data.html = marked.parser(vueCodeTokens);  return `  <template>    <Result />    <hr />    ${data.html}  </template>  <script>import { defineComponent } from "vue";import Result from "${virtualPath}";export default defineComponent({  name: "ArgoDemo",  components: {    Result  }});</script>`;};
// utils.tsexport const isVirtualModule = (id: string) => {  return //@virtual/.test(id);};
export default function docPlugin(): Plugin {  let vuePlugin: Plugin | undefined;  return {    name: "vite:plugin-doc",    resolveId(id) {      if (isVirtualModule(id)) {        return id;      }      return null;    },    load(id) {      // 遇到虛擬md模塊,直接返回緩存的內容      if (isVirtualModule(id)) {        return getDescriptor(id);      }      return null;    },    async transform(code, id) {      if (!id.endsWith(".md")) {        return null;      }      if (isVirtualModule(id)) {        return await vuePlugin.transform?.call(this, code, getVueId(id));      }      // 省略其它代碼...    }  }}
// cache.tsconst cache = new Map();export const createDescriptor = (id: string, content: string) => {  cache.set(id, content);};export const getDescriptor = (id: string) => {  return cache.get(id);};

最后為示例代碼加上樣式。安裝prismjs

yarn add prismjs
// marked.tsimport Prism from "prismjs";import loadLanguages from "prismjs/components/index.js";const languages = ["shell", "js", "ts", "jsx", "tsx", "less", "diff"];loadLanguages(languages);marked.setOptions({  highlight(    code: string,    lang: string,    callback?: (error: any, code?: string) => void  ): string | void {    if (languages.includes(lang)) {      return Prism.highlight(code, Prism.languages[lang], lang);    }    return Prism.highlight(code, Prism.languages.html, "html");  },});

項目入口引入css

// main.tsimport "prismjs/themes/prism.css";

重啟預覽,以上就是vite-plugin-vue-docs的核心部分了。

遺留問題

最后回到上文構建組件style/index.ts遺留的問題,index.ts的內容很簡單,即引入組件樣式。

import "../../style/index.less"; // 全局樣式import "./index.less"; // 組件樣式復制代碼

index.ts在經過vite的lib模式構建后,我們增加css插件,在generateBundle鉤子中,我們可以對最終的bundle進行新增,刪除或修改。通過調用插件上下文中emitFile方法,為我們額外生成用于引入css樣式的css.js。

import type { Plugin } from "vite";import { OutputChunk } from "rollup";export default function cssjsPlugin(): Plugin {  return {    name: "vite:cssjs",    async generateBundle(outputOptions, bundle) {      for (const filename of Object.keys(bundle)) {        const chunk = bundle[filename] as OutputChunk;        this.emitFile({          type: "asset",          fileName: filename.replace("index.js", "css.js"),          source: chunk.code.replace(/.less/g, ".css")        });      }    }  };}

結語

下篇暫定介紹版本發(fā)布,部署站點,集成到在線編輯器,架構復用等,技術涉及l(fā)inux云服務器,站點服務器nginx,docker,stackblitz等。

(學習視頻分享:vuejs入門教程、編程基礎視頻)

以上就是【由淺入深】vue組件庫實戰(zhàn)開發(fā)總結分享的詳細內容,更多請關注php中文網其它相關文章!

關鍵詞: Vue.js Vite

男女啪啪高清网站| 亚洲最大黄色福利| 亚洲欧美日韩激情视频| 欧美精品在线视频| 中文字幕丝袜美腿诱惑色在线观看| 亚洲av超清在线| 伊人久久视频在线观看| 精品91久久99九九| 亚洲视频亚洲视频| 日韩av一区二区三区人妻| 中文字幕 日韩观看视频| 精品日韩视频在线观看 | 超碰在线观看97视频| 中文字幕亚洲精品字幕| 国产日韩欧美亚洲91| 日韩无人区一卡2卡3卡4卡| 人人妻人人澡人人爽人人a v | 亚洲天堂网最新网址| 亚洲精品无码中文字幕无码| www.狠狠视频.com| 日韩免费观看v.| 久久伊人青青草大香蕉| 手机av永久免费| 久久久成人性生活小视| 老鸭窝三区免费在线观看| 视频一区二区三区免费观看| 国产探花熟女av在线| 就爱操在线视频观看| 91在线精品在线| 日韩午夜激情av| 91精品翘臀久久久| 黄版短视频在线观看| 成人在线免费观看久久视频| 欧美三级免费观看一区二区| 人人妻人人澡人人爽人人精品直播| 亚洲高清中文字幕不卡av一页| 日韩不卡视频一区| aaaaa青青草大片| 黄a大片av永久免费| 亚洲一区二区av在线网站| 丰满人妻一区二区免费视频| 日本女优在线三区| 一区二区三区欧美日韩| 精品视频久久一区二区三区四区| 精品久久久久久国产视频| 亚洲成人激情av在线播放| 97人妻免费视频精品| 国产精品久久久亚洲av| 国产香蕉久久精品免费| 欧美精品久久久久久久白浆| 91碰碰成人免费视频| 亚洲免费看125个视频| 人妻中文字幕精品| 日韩国产欧美激情在线视频| 999精品视频在线免费九九看| 亚洲欧美精彩视频| 亚洲Av无码成人黄网站在线| 直接看的av天堂| 国产一区二区精美视频| 免费看黄色的网站91| 超碰97免费公开在线人妻| 91影院免费破解污在线观看| 青青草免费国产视频| 一区二区三区四区丝袜| 国产亚洲成人av看黄在线观看| 精品一区二区三区 蜜臀av| 国产黄色一级黄色大片| 青青草国产在线免费观看 | 白嫩丰满少妇大战黑人| 精品在线激情av| 色婷婷视频免费在线| 超碰人妻人人爱人人干| 久久国语露脸精品国产麻豆| 免费观看日韩一级黄色大片| 自拍偷拍亚洲天堂精品| 精品久久97观看在线ship| 久久久久久久久久久女人拉屎α片| 欧美另类极品videoshd| 色哟哟在线观看永久免费视频| 日本冢本中文字幕| 97碰碰碰免费公开在线视频| 中文字幕中文字幕中文字幕| av中文字幕在线观看中文| 黄色片网站国产精品| 亚洲激情欧美伦理| 熟妇在线精品视频网站| 五月婷婷高清在线观看| 亚洲 欧美 另类 丝袜| 日韩av中文字幕在线观看地址| 一区二区三区内射少妇| 91狠狠人妻久久久久综合| 日韩中文字幕在线第一页| 亚洲视频精品一区二区三区四区 | 人妻人妻1区2区| 自拍视频在线观看一区| 97超碰在线高清| 久久熟妇一区二区三区| 黄色一级成人大片| 视色视频成人午夜精品| 人妻字幕av一区二区三区| 91在线视频亚洲| 超碰caoporn免费| 亚洲第一页色自拍| 日韩一级片在线视频观看| 二区三区精品在线观看| 丝袜人妻一区二区三区网站| 亚洲精品人成网址| 国产视频1区二区三区| 91国产老熟女精品| 妻子和黑人邻居的交换| 熟女口爆吞精合集| 国产欧美高清视频在线| 伊人午夜综合在线观看| 伊人伊人伊人伊人网| 伊人五月婷婷激情网| 另类专区亚洲欧美| 欧美三级免费观看一区二区| 久久国产成人精品免费视频| 23久久精品中文字幕人妻| 精品人妻日日夜夜操| 欧美 成人 一区 二区| 亚洲国产日韩不卡| 日韩一区二区在线观看视频| 国产乱老熟视频乱老熟女国| 久久精品视频在线看15| 91在线免费观看视频在线观看| 国产懂色av熟女丝袜精品| 国产亚洲精品美女久久久| 麻豆一区二区在线视频| 免费高清理伦片在线播放视频| 99视频在线国产观看| 在线欧美日韩观看一区二区| 五月婷婷中文字幕网| 99熟女精品一区二区三区| 亚洲第一页色自拍| 国产区在线观看视频全网yin乱| 亚洲性色av网站| 午夜在线视频播放网站| 国产av熟女一区二区三区四季| 色综合久久综合久久综合网| 国产麻豆精品成人av| 亚洲av中文字幕| 亚洲熟女一区二区三区电影| 欧美中文字幕精品在线观看| 五月婷婷综合中文字幕| 最新成人中文字幕| 日本久久久久久久国产精品| 久草视频在线有精品| 午夜精品av在线| 久久亚洲欧美日本精品品| 99热这里只有精品在线| 美女黄页网站久久久久久| 99视频30精品在线观看| 婷婷六月亚洲综合| 欧美极品激情一区二区三区 | 夜夜嗨av老熟女一区二区三区| 日韩中文乱码字幕| 亚洲Av无码成人黄网站在线| 99久久精品国产亚洲av热热爱| 97特黄一级中文字幕在线| 日本大胆女人视频| 午夜美女免费福利| 精彩视频久久久久| 999精品视频在线免费九九看| 不卡av中文字幕手机看| 五月婷婷色激情网| 久久精品国产精品国产精品国产 | 亚洲av乱码一区二区三区蜜臀| 999久久精品视频在线| 超碰在线观看97视频| 日韩亚洲av成人| 鸡巴插洞穴的软件免费试看| 神马伊人网久久久久久久久久久| 国产91精品福利在线观看| 日日天天日天天谢天天日| 99久久精品国产亚洲av热热爱| 自拍偷拍国产在线| 熟女口爆吞精合集| 中文字幕亚洲素人| 午夜精品久久久久久久久久久久久| 日韩美女精品视频| 成人理论片中文字幕| 国产精品久久久久久人妻爽| 人妻 中文字幕 森泽佳奈| 超级碰人妻香蕉在线97| 亚洲三级激情在线观看| 无码人妻丰满熟妇区五十路| 欧美亚洲第28页| 黄色片网站国产精品| 女人的天堂av网| 国产成人看片在线| 五月婷婷久久中文| av少妇人妻中文字幕| 夜夜嗨一区二区三区直播内容| 国产日韩欧美一区激情| 日韩毛片短视频在线播放| 国产精品久久久精品久久| 婷婷在线www.| 亚洲欧美日韩激情视频| 大香蕉伊人久久草| 伊人久久视频在线播放| 国产视频一区 在线播放| 五月婷婷色激情网| 丝袜人妻诱惑在线| 人妻超碰在线观看| 日本乱码视频在线播放| 99永久免费精品视频| 成人动漫在线观看播放| 精品免费污污网站在线观看| 人妻中文字幕不卡| 2020国内精品免费视频| 美女网站视频大全| 天天舔,天天做,天天爽| 亚洲一区天堂在线| 日本在线高清视频| 久久久综合网久久久综合网| 高清一区二区三区免费| 天天射天天干天天舔| 日韩中文字幕在线观看乱码| 无人妻一区二区三区费中文字幕| 天天操美女综合网| 黄页网站网址在线观看| 视频一区视频二区中文精品 | 国产亚洲精品首页在线播放| 在线播放偷拍视频| 久久精品人人做人人综合| yeye38亚洲综合网久久| www.狠狠视频.com| 亚洲福利视频一区二区| 88人妻x88av| 激情欧美一区二区三区| 天堂性视频99在线播放免费| 国产网址视频在线观看| 精品毛片av一区二区三区| 久久99蜜桃精品视频| 日韩一区二区在线观看视频| 激情五月天在线观看色| 免费黄页网址大全在线观看| 日本爱片在线观看| 久久九精品综合丝袜影视精品| 日韩永久免费av网站| 精品亚洲精品国产一本大道| 亚洲成人天堂久久| 午夜激情福利大片| 东京热制服人妻诱惑| 青青青青久久精品国产首| 亚洲av色图网站| 麻豆成人性电影在线观看视频| 高清一区二区三区免费| 国产99精品久久久久久久| 色婷婷综合久久久精品视频| 久久精品亚洲国产av香蕉| 蜜桃一区二区三区在线观看视| 婷婷综合在线五月天| 岛国黄色在线网站| 国产一区二区在线播| 国产熟女啪啪精品一区二区三区| 精品亚洲精品国产一本大道| 2020午夜免费福利| 91精品国产丝袜长腿久久| 欧美日韩亚洲中文色| 中文人妻精品一区二区三区四区| 国产懂色av熟女丝袜精品| 91国产老熟女精品| 97人妻免费视频精品| 亚洲一区二区 偷拍| 午夜精品视频在线观看视频| 久久精品国产亚洲av五区| 粉嫩一区二区性色粉嫩av| 伊人久久视频在线观看| 日韩熟女在线视频| 日本aaaaa级特黄大片老头| 中文字幕 日韩观看视频| 999久久久精品国产| 久久国产精品亚洲欧美阿娇| 国产又粗又长的视频| 老熟妇高潮一区二区三| 99亚洲国产精品久久久| 日韩 欧美 国产 丝袜| 77777蜜臀精品久久综合| 精品丰满少妇人妻| 国产精品免费自拍视频大全 | 亚洲视频亚洲视频| 欧美日韩视频高清一区二区| 国产伦精一区二区三区| 亚洲av少妇高潮150p| 大香蕉伊人久久草| 日本老女人性生活视频| 国产又粗又硬又爽又黄的网站| 99r精品视频在线播放| 国产精品一区二区三区色| 久久国产乱子伦免费看| 日韩av在线激情| 久久久久久亚洲精品中文字幕| 少妇毛片一区二区三区| 国产麻豆激情图区在线| 成人精品永久免费观看| 欧美风情日韩国产黑白配| 日韩中文字幕激情在线| 亚洲av操伊人久久| 亚洲一区二区人妻| 日韩毛片免费播放| avove在线视频| 天天操天天爽天天舔天天射| 久久久久久久久久一级| 国产一区区二区三区| 日本成人中出视频| 国内精品久久久久久精品视频| 中文字幕 一区二区| 社区福利一区二区三区| 99视频在线国产观看| 午夜亚洲激情久久| 亚洲五月六月激情| 最全同人动漫网址| 日韩精品少妇熟女| 亚洲欧美国产麻豆综合| 97在线视频在线激情| 久久亚洲中文字幕精品| 夜夜嗨av老熟女一区二区三区| 精品黄色美女一二区| 精品国产亚洲蜜月在线观看| 日韩毛片成人在线| 超碰在线97国产久中文| 精品丰满少妇人妻| 精品一区二区三区不卡蜜臂| 亚洲av日韩四区| 日韩电影亚洲专区天堂| 97在线观看国产免费视频| 久久久久久久久久亚洲影视| 国产精品人妻在线| 久久资源网国产精品| 色爱综合久久一区二区三区蜜桃臀| 99热精品在线免费观看| 日韩亚洲av成人| 99精品欧美一区二区三区蜜桃| 亚洲熟女一区二区三区电影| 精品少妇高潮涩涩av| 亚洲国产婷婷六月丁香伊| 亚洲成人激情av在线播放| 久久激情欧美在线播放| 青青青青久久精品国产首| 97碰碰碰免费公开在线视频| 精精品人妻一区二区| 青青青青久久精品国产首| 手机在线日韩av| 中文字幕色123| 69视频在线 91| 性高潮久久久久久久久免费| 日本最新在线不卡免a费视频| 中文字幕av人妻呻吟| 欧美日韩色图一区二区| 少妇精品久久综合网| 国产精品美女视频免费线播放| 黄色国产一区在线观看| 国产综合一区久久91| 红桃在线观看av| 欧美经典一区二区三区| 精品亚洲午夜久久久久四季| 精品欧美日韩免费| 超碰免费在线不卡不卡不卡| 亚洲免费观看毛片| 红桃在线观看av| 亚洲岛国av在线| 91精品人妻呻吟| 丝袜人妻av中文字幕| 久久久久久久久久久久少妇| 一区二区三区四区丝袜| eeuss一区二区人妻| 清纯唯美亚洲五月激情综合| 亚洲不卡免费在线| 亚洲精品人成网址| 色香视频亚洲自拍偷拍| 九九热播这里只有精品| 日韩国产欧美三级| 精品亚洲精品国产一本大道| 夫妻性生活一级黄色大片| 久久久无码av一区二区三区| 色婷婷狠狠18禁久久久| 天堂性视频99在线播放免费| www.免费成人在线视频| 操美女姐姐啊啊啊| 免费黄免费在线观看| 日韩女人小便视频| 国产欧美1区2区3区| 国产精品亚洲欧美综合| 色婷婷久久综合久色| 无码人妻丰满熟妇区五十路| 亚洲欧美在线第一页| 天天操天天射天天干天天爱| 超碰激情免费在线播放| yeye38亚洲综合网久久| 天天操天天透天天射| 午夜蜜臀一区二区三区四区五区 | 久久成年人免费网站| 激情久久bbbwww| 日韩毛片短视频在线播放| 麻烦视频一区二区三区| 国产精品va一级二级三级| 日韩av永久免费观看| 52av成人久久国产一区| 青青在线观看视频精品| 一本之道久久久久久久| 啄木乌av一区二区三区| 丁香花中文字幕在线观看| 中文字幕av在线更新| 亚洲风情资源网站| 黄片激情在线观看| 麻豆av熟女少妇精品| 日韩毛片av网站| 天天操天天透天天射| 超碰在线97国产久中文| 一区二区三区欧美日韩| 日本免费久久精品| 久久91精品国产91久久蜜月 | 色狠狠亚洲爱综合国产| 日本人妻乱子免费播放| 精品人妻一区二区区中文| 91精品国产一二三产区区别在哪| 97国产三级精品在线| 中文字幕人妻少妇一区二区| 日韩av网站 狠狠| 99re视频免费热线观看| 超碰超碰超碰在线免费观看| 免费观看视频成人| 亚洲丝袜中文字幕在线观看| 网页端在线聊天室| 丝袜美腿免费在线| 国产精品久久无码一区二区三区 | 亚洲婷婷91中文字幕| 日韩精品一线二线三线四线 | 淫妇天天干夜夜操| 2020国内精品免费视频| 欧美日韩中文字母专区| 日本伊人网在线播放| 国产精品黄视频免费看| 91国语对白精品露脸| 91精品国产综久久久久| 婷婷91人妻精品一区二区三区| 日韩av俩男操女人吃奶| 人妻丰满熟妇av一区二区| 色综合久综合久久综合久鬼88| 99自拍视频国产在线| 日本 韩国 欧美 久久久| 午夜爽视频在线观看| 亚洲一本一道一区二区三区| 欧洲免费无线码在线一区| 一区二区电影成人| 久久九九在线观看视频三级| 日韩精品色图在线| 日本欧美三级高潮受不了| 中文字幕天堂av.| 欧美日韩亚洲成人一区| 91精品国产91久久久久杨7| 日韩精品毛片免费播放蜜臀| 99手机视频免费在线观看| 久操在线免费观看视频| 日韩一级av电影| 一区二区电影成人| 日韩高清av在线免费观看| 日韩 欧美 国产 丝袜| 亚洲av操伊人久久| 91人妻人人看人人爽| 青青在线观看视频精品| 久久66热人妻偷产精品9| 午夜美女免费福利| 91性高久久久久久久久| 久久97精品久久久久久久不卡 | 亚洲国产精品成人久久久| 波霸肥熟女bbw| 自拍视频在线观看一区| 欧美日韩国产激情不卡| 亚洲av成人一区午夜网站| 亚洲超爽美女毛片| 日本女优在线三区| 欧美日韩亚洲中文色| 午夜美女诱惑福利| 日韩欧美在线观看二区| 视频一区二区三区免费观看| 在线观看视频瑟瑟| 亚洲综合一区在线| 在线观看日韩黄色蜜桃| 一区二区三区四区丝袜| 91一区二区久久国产乱| 久久综合日日夜夜| 人人妻人人澡人人爽人人a v| 丰满人妻一区二区免费视频| 成人av在线天堂一区二区三区| 男人操女人逼的免费视频| 日本高清久久久久久久| 国产九色91在线视频| 亚洲美女天堂av| 黑人上司魅惑人妻电影| 国产蜜桃av在线观看| 精品视频久久一区二区三区四区| av网站软件在线播放| 青青操视频在线观看免费观看| 蜜桃av一区二区精品| 国产精品毛片久久久久久久久| 国产91熟女高潮一区二区| 中文字幕不卡一区二区三区| 精品噜噜噜噜久久久久久久久试看| 最新精品国产精品1| 女同性恋69av| 国产欧美日韩精品一区二区| 亚洲av成人波多野一区二区| av毛片大全亚洲| 亚洲天堂黄色网络| 欧美精品一区二区久久不卡| 伊人久久视频在线观看| 91久久国语露脸精品国产| 亚洲精品无码中文字幕无码| 日韩午夜久久av| 午夜精品av在线| 中文字幕久久一二三四区| 中文字幕有码系列| 国产精品久久久久久av蜜臀 | 美女黄页网站久久久久久| 天天操天天爽天天舔天天射| 日韩永久免费av网站| 蜜臀国产综合久久第一页| 中文乱码人妻一区二区三区| 日本人妻乱子免费播放| 精品久久久久久国产视频| 在线97视频观看| 人人妻人人爱人人澡| 99久热er在线精品视频| 日韩中文av在线观看| 国产日韩高清一区二区| 91诱惑极品美女| 亚洲视频区1000| 国产精品久久国产精品99- | 久久aa黄色aa网站| 精彩视频久久久久| 日韩 欧美 中文在线| 中文字幕日韩一二三区视频| 日韩精品中文字幕巨臀人妻中出| 色综合色综合网站| 伦理福利视频导航| 国产91熟女被艹嗷嗷叫| 丝袜人妻精品一区二区三区| 国产亚洲欧美日韩国产精品三区| 精品欧美日韩免费| 91在线精品在线| 日韩 欧美成人999| 久久国产乱子伦免费看| 福利无码一区二区三区| 婷婷亚洲综合久久| 一级黄色片录像片| 熟女人妻在线视频第一页| ff14一区二区三区分别是啥| 亚洲Av无码成人黄网站在线| 日韩在线中文字幕在线| 久久久久国产精品黄毛片| 亚洲男人的天堂久久| 国产精品一区二区流白浆| 国语自产拍在线观看视频| 午夜在线视频播放网站| av丰满激情人妻在线| 亚洲欧美熟女一区二区三区| 亚洲精品久久久久综合| 自拍视频在线观看一区| 免费黄色特级大片| 久久综合亚洲狠狠伊人| 狠狠综合久久综合88亚洲| 亚洲毛片在线播放| 亚洲欧美日韩国产综合第一产区| 午夜蜜臀一区二区三区四区五区| 使劲点爽视频网站| 天天干天天综合色| www国产亚洲精品久久久| 日本女人干逼视频| 久久精品男人的天堂av蜜臀| 99亚洲国产精品久久久| 国产福利小视频在线观看免费 | 成人伊人精品色xxxx视频 | 人人妻人人澡人人爽人人a v| 天天插天天日天天啪| 青青草视频 成人| 亚洲欧美日韩一区另类| 老司机av在线视频| 亚洲福利视频一区二区| 国产福利小视频在线观看免费| 国产麻豆激情图区在线| 2020国内精品免费视频| 丁香欧美欧美欧美欧美欧美| 亚洲成人人妻一区| 久久精品国产露脸对白| 亚洲精品又黄又爽的视频| 久久日韩美女人妻精品| 精品人妻交换视频在线看| 最新久久精品免费视频| 国产精品欧美一区二区三区不卡| 国产视频欧美一区二区| 激情欧美在线激情| 国产在线av小视频| 色香视频亚洲自拍偷拍| 日韩精品成人在线免费观看| av网站国产在线| 麻豆国产在线观看视频| 日韩午夜福利视频合集| 国产av在线播放网站| 粉嫩av在线综合| 中文字幕久久中文字幕综合网| 狠狠挺进h嗯啊第一次视频| 深夜视频在线四区| 亚洲综合精品伊人久久| 国产成人午夜三级在线观看| 100xxoo蜜乳| 国产欧美日韩精品一区二区| 精品久久在精品久久| 精品视频卡一卡二卡三卡四卡| 久久成人av一区二区| 一本色道久久爱久| 日韩母乳女优激情视频在线观看| 午夜蜜臀一区二区三区四区五区| 日韩毛片av网站| 少妇被爽到高潮喷水久久夜色| 911美女片黄在线观看| 99视频30精品在线观看| 久久少妇高潮视频免费| 人人妻人人澡人人爽人人精品直播| av少妇人妻中文字幕| 成人黄色在线电影网站| 亚洲视频亚洲视频| 久久久久久久久久久东京| 2019中文字幕免费在线观看| 久久人人妻人人妻人人澡av| 成人黄色性a大片| 日韩一级av电影| 日韩中文字幕综合在线网| 久久 久久 久久av| 国产精品高潮呻呤久久av| 果冻亚洲国产成人av播| 国产伦精品一区二区三区视频9| 美女福利视频午夜| 色哟哟在线观看永久免费视频 | 蜜桃一区二区三区在线观看视| 国产一区二区三区很黄很爽| 亚洲欧洲av天堂| 中文字幕av人妻呻吟| 久久精品国产亚洲avapp| 午夜av网站观看| 91精品国产综久久久久| 深夜美女福利诱惑| 亚洲三级激情在线观看| 亚洲三级黄色av| 成人黄色在线电影网站| 日韩色图欧美视频| 亚洲三级av一区| 一级片一级黄色片| 免费人人潮人人爽一区二区| 亚洲国产婷婷六月丁香伊| 激情av一区二区三| 美女小视频在线观看| 日本女人干逼视频| 国产视频97在线播放| 日韩美女毛片午夜剧场| 婷婷综合在线五月天| 丝袜人妻精品一区二区三区| 国产av在线播放网站| 无人一区二区三区在线观看视频 | 97久久香蕉国产线看观看| 久久亚洲中文字幕精品| 99热只有国产在线精品| 国产精品久久久久精品三级a| 亚洲一区二区 偷拍| av资源在线一区二区三区| 天天色天天干网址| 中文字幕乱码高清视频在线| 成人黄色在线电影网站| 日韩亚洲欧美在线第一| 免费高清理伦片在线播放视频 | 一区二区人妻字幕| 日韩伦理不卡一区二区| 99视频这里都是精品6| 国产精品亚洲av三区| 亚洲人成精品久久久久| a在线视频播放免费网站| 美女激情国产精品| 婷婷亚洲综合久久| 国产三级精品久久久| 国产精品综合久久桃花| 亚洲性色av网站| re久久99热国产| 久久久蜜桃成人网| 日韩一级av电影| 午夜激情成人在线| 久久婷婷视频一二三区| 久久一区二区三区av| 成人av电影网站日韩| 少妇人妻在线一区二区| 高潮喷水在线欧美| 日韩在线中文字幕在线| 国产亚洲成人av看黄在线观看| 亚洲 欧美 另类 丝袜| 最新福利网址91| 一区二区三区国产日本欧美| 麻豆成人久久精品一区二区三区| 久久久成人性生活小视| 全网小视频资源免费在线播放| 人妻精品一区二区久久| 青青青在线视频人视频| 日本影片高清视频| 美女色网站在线不卡粉嫩av| 天天舔,天天做,天天爽| 亚洲天堂黄色网络| 青青操在观看视频| 欧美日韩在线视频三区| 国产精品高潮呻呤久久av| 亚洲精品9999久久久久| 97国产三级精品在线| 精品国产久久免费观看| 久久久精品国产sm调教| avove在线视频| 视频一区二区三区免费观看| 久久综合亚洲狠狠伊人| 欧美成人精品三级在线| 欧美日韩亚洲中文色| 最新日韩在线观看视频| 亚洲一卡2卡三卡| 亚洲美女巨乳在线| 五月婷婷六月在线观看| 欧美mv日韩mv视频| 日韩母乳女优激情视频在线观看| 23久久精品中文字幕人妻| 中文字幕人妻少妇一区二区| 精品人妻三区日日| 日韩精品亚洲性图欧美| 香蕉成人在线91| 免费中文字幕啪啪啪| 午夜熟女经典一区二区| 五月婷婷欧美一区| 日本视频一区免费| 1024人妻熟女一区二区三区| 久久偷拍美女上厕所| 青青手机国产视频| 天天色天天黄天天操天天射| 999精品视频在线观看精品| 88成人美女女内射| 成人国产av免费视频下载| 亚洲av懂色av| 亚洲精品无码中文字幕无码| 4438x亚洲最大成人网| 国产一区二区三区香蕉| 96国产av传媒精品| 国产精品久久久久久久久果冻传媒| 国产精品久久久久精品三级a| 精品噜噜噜噜久久久久久久久试看 | 精品两个孩子一区二区三区| 91人妻人人看人人爽| av网站软件在线播放| 四虎av在线观看| 日韩欧美高清一区二区三区| 91精品人妻麻豆| 玩弄丰满人妻一区二区av| 天天插天天日天天啪| 亚洲一区成人动漫在线观看| 国产裸体一二区三区视频| 亚洲欧美在线第一页| 有码中文av字幕| 亚洲精品高清一二| av天堂地址在线观看| 国产熟妇另类久久久久久| 国产裸体学生视频全黄网站| 欧美黄色aaa级| 日韩在线一区二区三区精品| 欧美日韩在线视频在线| 无人区午夜精品乱码一区二区| 国产精品欧美日韩在线观看一区| 97视频公开在线观看| 色婷婷五月亚洲综合小说| 在线观看日韩黄色蜜桃| 亚洲av少妇高潮150p| 男人天堂成人亚洲| 超碰在线成人97| 最近最好的中文字幕免费| 日韩人妻激情瑟瑟| 四虎av在线观看| 99精品欧美一区二区三区蜜桃| 国产综合一区久久91| 人妻乱干视频免费| 超碰成人97在线| 国产情趣丝袜高跟av色| 久久久久久久久久久久少妇| 99亚洲综合精品成人网色播| 偷拍 自拍 在线| 偷拍99免费视频| 网页端在线聊天室| 大量老熟女偷拍视频老女人| 国内美女直播视频| 亚洲情色伦理在线| 人妻 制服 日韩 中文 在线| 精品欧美日韩免费| 国产精品毛片久久久久久久久| 欧美二区三区精品在线| 91久精品人妻中文一区二区| av影音在线不卡| 日本aaaaa级特黄大片老头| 欧美激情1区二区三区| 国产一区二区在线播| 亚洲成网在线观看| 日韩深喉口爆吞精| 日韩欧美一区二区在线免费观看 | 男人日女人的逼的视频| 午夜蜜臀一区二区三区四区五区| 欧美日韩在线视频第三区| 狠狠操狠狠干狠狠操狠狠干| 色爱综合久久一区二区三区蜜桃臀 | 夫妻性生活一级黄色大片| 亚洲一区日韩三级| 奶头被吸得又大又黑np| 人妻人伦精品国产| 99视频这里都是精品6| 91国语对白精品露脸| 国产av熟女一区二区三区四季| 欧美精品一区二区免费开放| 亚洲情色 一区二区三区| 国产一区区二区三区| 日韩av另类在线| 国产精品色悠悠在线观看| 国在线产视频91高清| 婷婷亚洲综合久久| 亚洲五月六月激情| 久久精品国产亚洲av麻豆软| av网站大全在线免费观看| 国产一区二区av少妇| 自拍另类亚洲欧美| 欧美中文字幕精品在线观看| 亚洲国产成人精品女人久久久小说| 神马一区二区三区伦理片| 精品人妻一区二区免费| 91热视频在线观看| 国产超碰人人一区二区三区| 亚洲欧美国产免费| 一区二区三区在线视频精品| 亚洲天堂成人网入口| 美女视频黄免费的亚洲男人天堂| 伊人午夜综合在线观看| 肥屁股高潮迭起一区二区的视频| 凹凸国产熟女白浆精品视频免费| 天天操天天爽天天舔天天射| 日本女优与黑人特级| 国产自拍偷拍日韩精品| 欧美黑人一区二区在线| 亚洲一区日韩三级| 久久久九九九精品视频| 国产人与禽zoz0性情伦| 精品人妻少妇久久久久久| 国产97综合久久久| 久久婷婷视频一二三区| 欧美极品激情一区二区三区| 国产麻豆激情图区在线| 午夜亚洲激情久久| 日韩丝袜人妻中文字幕| 狠狠操狠狠干狠狠操狠狠干| 自拍伦理视频在线观看| 成人一级视频在线播放| 青青青在线视频人视频| 亚洲女人的av天堂| ff14一区二区三区分别是啥| 出轨少妇自白小说| 人人妻人人澡人人爽人人精品直播| 偷拍99免费视频| 中文字幕在线三级| 国产精品久久久久久人妻爽| 黑人巨大的吊bdsm| 99久久国产综合精品久久国产 | poronovideos美女| 中文字幕日韩精品在线免费| 国产精品人妻在线| 三级网站久久综合| 91精品国产综合久久香蕉第1集 | 日韩 午夜 免费| 日韩中文字幕在线观看乱码| 精品高清亚洲国产欧美| 伊人av在线播放| av在线免费观看天堂| 午夜美女免费福利| 亚洲另类伦春色综合小| 超碰成人97在线| 中文字幕人妻网站| 99视频30精品在线观看| 俺去啦视频在线精品| 和大屁股女人臀交| 亚洲精品又黄又爽的视频| 亚洲欧美日韩丝袜色图| 亚洲综合精品伊人久久| weyvv5国产成人精品的视频| 久久精品国产露脸对白| 欧美日韩在线视频第三区| 中文在线字幕丝袜美腿| 日本老女人性视频| 国产伦理精品av| 欧美成人金8天国加勒比| 午夜免费福利视频| 亚洲激情欧美伦理| 热久久77这里有精品综合久久| 国产91熟女高潮一区二区| 精品人妻一区二区三区四在五区| 高清在线 中文字幕| 国产人与禽zoz0性情伦| 国产人伦人妻亚洲| 日本老女人性生活视频| 农村人妻一区二区三区| 狠狠操你在线观看| 亚洲经典 中文字幕| 大香伊蕉人在人在线国产精品 | 激情亚洲精品在线观看| 国产 日韩 精品在线观看| 经典a级片在线观看| 免费高清理伦片在线播放视频| 黑人巨大欧美一区二区视频| 中文字幕亚洲综合色| 日韩人妻精品一二三| 日韩av永久免费观看| 国产黄色一级黄色大片| 91亚洲视频在线观看黄| 青青操操操操操操操| 天天综合天天精品| 成人校园春色小说| 美女黄页网站久久久久久| 蜜桃成人永久免费av大| 亚洲午夜丝袜诱惑| 日韩蜜臀人妻素人精品在线观看 | 粉嫩一区二区性色粉嫩av| 久久久久久精品国产婷婷| 久久婷婷视频一二三区| 国产熟女一区二区91| 欧美情色免费视频| 欧美日韩激情文学| 九九热九九色九九操| 在线中文字幕在线中文| 国产 欧美 日韩在线视频| 在线播放日韩精品av| 一级国产黄片国语对白| 久久国语露脸精品国产麻豆| 久久午夜一成年人的视频| 人妻视频在线免费播放| 亚洲人妻一区二区在线观看| 亚洲性感天堂欧美| av在线播放观看播放观看| 曰曰摸夜夜添av老司机蜜桃视频| 18禁涩涩网站在线| 亚洲国产日韩不卡| 青青草原国产在线精品| 91最新视频在线播放| 人妻色图欧美日韩| 丝袜人妻诱惑在线| 精彩av在线不卡播放| 日韩av在线大片| 精品少妇高潮涩涩av| www.99久久久久.com| av男人的东京天堂热| 五月婷婷综合激情五月| 清纯唯美亚洲五月激情综合| 免费看插b视频网站| 欧美日韩1区2区3区4区5区| 国产一区二区在线校花| 亚洲麻豆综合精品| 91久精品人妻中文一区二区| 久久精品国产精品国产精品国产| 亚洲性色av网站| 麻豆成人久久精品一区二区三区| 99国内外在线视频免费| 91小情趣制服肥臀九色| 午夜色网av在线| 草原免费视频亚洲| 谁有av网站在线播放中文字幕| 超碰人妻人人爱人人干| av全国在线观看| 精品毛片av一区二区三区| 五十路熟女人妻在线网观看| av最新在线中文天堂| 亚洲无av码一区二区三区| 丝袜 成人 av| 香蕉国产精品久久| 久久久久久久久久一二三| 亚洲午夜丝袜诱惑| 国产av专区网站大全| 免费一区二区风骚徐娘| 国产又粗又猛又爽又黄的视频在线| 中文字幕无线码在线观看| 天天摸天天摸天天摸| 亚洲高清中文字幕不卡av一页| 青青操在观看视频| 亚洲av黄久久久| 成人精品永久免费观看| 最新亚洲国产高清激情| 中文字幕无线码在线观看| 最新人妻激情视频网| 老熟妇高潮一区二区三| 日韩免费黄色在线| 青青手机国产视频| 在线亚洲无av码| 国产一区二区三区女人的床叫声| 巨乳中文字幕一区| 91一区二区三区在线视频| 国产卡一卡二专区| av在线免费高清国语| 午夜精品在线观看成人| 久久久久久精品免费免费sss| 国产黄色剧情影片麻豆免费播放| 亚洲成在人线视av| 激情综合网日韩av| 99视频30精品在线观看| 久久精品国产亚洲av久| 夜夜嗨av老熟女一区二区三区| 把高跟丝袜美腿扛在肩上| 熟女人妻中文字幕一三区| 亚洲精选国产一区| 亚洲男人的天堂色偷免费| 中文字幕久久j日| av网站国产在线| 久久精品国产亚洲激激情| 亚洲欧美日韩综合在线丁香| 久久久久久 九色av| 手机在线日韩av| 日韩中文字幕一区二区高清| 人妻熟女中文av| www.免费成人在线视频| 日韩va欧美激情在线| 人妻熟女中文av| 久久久综合网久久久综合网| 大量老熟女偷拍视频老女人| 2020午夜免费福利| 日本老熟妇ⅹxx| 香蕉国产精品久久| 亚洲综合日韩另类| 亚洲成天堂人动漫| 91精品国产91久久久久杨7| 色av色婷婷18人妻久久久| 强伦人妻一区二区三区视频18| 2020午夜免费福利| 另类日韩一区二区三区| 一级特色特级黄大片| 欧美日韩色图一区二区| 日韩va欧美激情在线| 亚洲乱码久久中文| 久久精品国产露脸对白| 国产精品视频福利在线| 日韩亚洲欧美在线第一| 瑟瑟视频免费观看在线| 人妻 在线 二区| 尹人香蕉久久99天天拍久女久| 又黄在线免费观看视频| 国产精品999动漫| 亚洲国产精选视频在线观看| 日韩最新视频播放| 高清在线 中文字幕| 中文字幕亚洲精品字幕| 四虎av在线观看| 日韩三级电影一区二区三区| 中文字幕日韩在线av| 少妇人妻在线一区二区| 久久精品国产露脸对白| 欧美一区二区三区免费的网址| 自拍偷拍国产在线| 国产熟女一本区三区四区| av人妻日本不卡二区| 网页端在线聊天室| 久久亚洲中文字幕精品| 激情五月婷婷综合色视频| 日韩国产欧美激情在线视频| 日韩99中文字幕在线视频| 欧美男人女人在线视频| 国产精品人妻系列23p| 无人区午夜精品乱码一区二区| 骚片av蜜桃精品一区| 日韩中文有码在线视频| 和大屁股女人臀交| 中文av字幕在线观看一区| 手机在线免费国产| 亚洲精品9999久久久久| 精产国品av一二三产区| 亚洲精品在线免费观看av| 激情综合网日韩av| 国产亚洲精品久久久一区二区| 午夜一级免费福利视频| 国产一区二区三区女人的床叫声| 日韩伦理在线观看一区视频| 中文字幕一级不卡| 婷婷激情网婷婷激情网| 久久午夜亚洲精品网站| 久久精品苍井空精品久久| 婷婷综合在线五月天| 色香视频亚洲自拍偷拍| yellow中文字幕视频| 老司机av在线视频| 四十路g五十路熟女豊满av| 精品两个孩子一区二区三区| 国产精品黑丝美腿美臀| 伊人午夜综合在线观看| 激情懂色av一区av二区av| 日韩一区二区在线观看视频| 国产自拍偷拍日韩精品| 久久精品av网站在线播放| 97国产一区二区三区| 99久久综合精品五月天| 色婷婷a区一区二区三区| 4438x中文字幕| 欧美熟妇大阴蒂高潮系列| 人人妻人人澡人人爽人人精品直播| 国产 欧美 日韩在线视频| 91熟女国产老熟女| 日韩av俩男操女人吃奶| 四虎影院一级黄色片| 999久久久精品国产| 日韩 欧美 国产 丝袜| 国产精品久久久久久久久av| av人妻精品一区二区三区| 日韩精品中文字幕日韩美| 99热只有国产在线精品| 婷婷只有在线精品视频| 亚洲国产自拍在线| 国产美女蜜臀av怡红| 草原免费视频亚洲| 欧美与黑人午夜交久久久| 91在线国产观看视频| 婷婷久久香蕉五月综合| 无人一区二区三区在线观看视频 | 亚洲成av人片一区二区久久久| 丝袜美腿免费在线| 国产又粗又爽又猛视频| 超碰人妻人人爱人人干| 99久热er在线精品视频| 日本aaaaa级特黄大片老头| 欧美日本一道本免费三区| 久久综合另类激情人妖| 国产一区二区在线激情欧美| 久久久久久久久久久女人拉屎α片| 色就是色综合欧美色| 亚洲熟女一区二区三区电影| 午夜一级免费福利视频| 看久了久久久久久久久久| 超碰成人97在线| 中文字幕久久一二三四区| 国产伦精品一区二区三区视频9| 亚洲色影在线视频| 亚洲三级激情在线观看| 午夜激情免费视频| 黄色录像一级片大| 超碰caoporn免费| 亚洲欧美日韩激情在线| 天天爽夜夜爽人妻va| 国内精品伊人久久久久av| 性感丝袜美女诱惑| 国产第一影院草草影院久久| 日韩av一区二区三区人妻| 视频在线观看色版| 自拍另类亚洲欧美| 玩弄丰满人妻一区二区av| 亚洲 欧美 中文字幕第一页| 日本中文字幕诱惑| 亚洲五月六月激情| 色综合久久中文综合久久| 久在线视频_老鸭窝在线视频| 使劲点爽视频网站| 美女视频黄免费的亚洲男人天堂| 久久久无码av一区二区三区| 日韩精品色图在线| 人妻人妻1区2区| 国产精品久久久亚洲av| 亚洲欧美自拍偷拍激情| av中文字幕乱码在线看| 在线中文字幕超碰| 日韩电影亚洲专区天堂| 91色综合综合热五月激情| 国产一区二区三区香蕉| 婷婷综合在线五月天| 自拍偷拍美腿丝袜亚洲| 久久久久久久久久久久久福利| 色就是色综合欧美色| 国产三级 日韩三级| 亚洲午夜久久精品视频| 免费看插b视频网站| 国产日韩高清一区二区| 亚洲女人的av天堂| 久久97精品久久久久久久不卡 | 色94色一区二区三区| 国内毛片精品一区二区三区| 激情五月天在线观看色| 日韩av网站 狠狠| av毛片大全亚洲| 无人区午夜精品乱码一区二区| 日韩精品成人在线免费观看| 日本高清中文字幕在线| 日本中文字幕在线观看免费| 自拍偷拍第四色偷拍| 黑人巨大欧美一区二区视频| 日一区二区三区在线视频| 亚洲中文资源在线| 国产激情怍爱视频在线| 精品国产91久久久久久久| 久久久久久精品成人| 久久久中文字幕人妻| 63久久av精品一区二区| 欧美精品一区二区久久不卡| 青青青草草草久草| 蜜臀久久99精品久久久兰草影| 一道久久久综合狠狠| 日本女优与黑人特级| 顶级少妇一区二区三区| 国产一区二区视频麻豆| 日韩中文字幕三区| 免费在线观看亚洲黄色| 粉嫩一区二区性色粉嫩av| 久久久精品国产人妻在线观看| 亚洲超爽美女毛片| 亚洲中文av天堂| 国产av高清无限看| 色婷婷五月亚洲大全| 视频在线观看色版| 国产日韩欧美三级在线| 中文字幕av最新更新| 欧美黑人一区二区在线| 婷婷久久综合久色综| 日韩啊v视频在线| 天堂男性av在线| 欧美日本一道本免费三区| 日韩成人午夜福利影院| 精品人妻一区二区免费| 亚洲中文字幕视频乱码| 黄色理论片在线观看| 国产精品99久久久久久天天| 色综合久久综合久久综合网| 骚熟妇人妻中文字幕网站| 国产女同一区二区久久| 天天色天天舔天天干| 午夜色网av在线| 尹人香蕉久久99天天拍久女久| 欧洲免费无线码在线一区| av网站国产在线| 久久久久久久久久一级| 亚洲美女高潮久久| 不卡av中文字幕手机看| 亚洲熟女av观看| 青青青草草草久草| 99久热er在线精品视频| 日韩综合人妻av| 亚洲中文av字幕综合| 有没有黄色一级片| 日韩欧美一区黄色| 亚洲av操伊人久久| 成人av欧美在线观看| 日本成人中出视频| 自拍偷拍另类图区| 国内精品久久99人妻无| 男女av在线观看网站| 青青电视剧全集免费观看| 麻豆一区免费爱爱视频| 88成人美女女内射| 中文字幕av三级免费| 亚洲国产精品成人久久久| 国产网址视频在线观看| 亚洲综合自拍成人偷拍网站| ysl蜜桃色7v| 中文字幕人妻久久不卡| 黑人巨大超大另类videos| 黄色录像一级片大| 午夜久久福利电影网| 国内精品伊人久久久久av| 免费观看性感美女| 免费人人潮人人爽一区二区| 美女内射白天91| 亚洲av男人的天堂久久精品| 日韩av不卡高清| 青青草原vip在线视频| 国产一级aa特黄大片| 99久久久久7777| 国产人妻精品中文字幕| 超碰人妻一区二区三区| 99国产精品久久久久久久成人| 综合亚洲婷婷小说| 在线 亚洲 精品| 黄色午夜免费网站| 亚洲春色男人天堂| 国产又粗又猛又爽又黄的视频在线| 日老熟女逼网视频导航| 国产日韩欧美一区激情| 肉色丝袜精品视频一区二区| 国产av 在线视频| 玩弄超骚少妇一区二区三区| 精品久久久久久久久99| 嫩草九九九精品乱码一二三| av大片在线看亚洲| 亚洲av影视综合| 狠狠婷婷久久精品一区二区| 日韩 欧美 丝袜| 欧美专区日韩精品一区二区| 小明看看成人播放平台| 偷拍视频你懂的一区二区三区| 亚洲伊人久久在线| 一本色道久久久亚洲天堂网| 无码人妻丰满熟妇区五十路| 搜索人妻av中文字幕| 久久久久人妻精品一区三寸| 日本中文字幕电影在线观看| 日韩在线一区二区三区精品| 在线播放偷拍视频| 日韩极品少妇的bbbbbb高潮| 亚洲一卡2卡三卡| 俺来也官网欧美久久精品| av在线免费观看天堂| 99视频这里都是精品6| 精品欧美日韩免费| 九九在线观看视频国产剧情| 偷拍自拍亚洲专区| 91精品视频在线观看免费版| 18成人黄色在线观看| 日韩在线中文字幕在线| 国产精品99久久久久久网站| 国产精品国产三级av| 亚洲天堂国产久久| 亚洲五月六月激情| 日韩毛片av网站| 一区三区精品视频在线观看| 999zyz玖玖资源站成人精品 | 暴露美女高潮喷水| 偷拍在线观看视频| 中国学习在线成人高考| 色香蕉av一区二区| 日本老女人性生活视频| 日韩性在线观看视频| 有码中文av字幕| av少妇人妻中文字幕| 天天看天天做天天色| 久久久精品国产人妻在线观看| 欧美三级中文字幕在线| 国产av一区二区三区成人| 日韩最新视频播放| 亚洲天堂黄色网络| 日韩欧美国产12区| 久久久午夜精品日本| 亚洲一区二区 偷拍| 亚洲成av人片一区二区久久久| 97精品视频在线观看免费| 国产精品综合不卡| 日韩国产欧美激情在线视频| 蜜桃福利视频一区二区| 国产精品久久人人添| 深夜美女福利诱惑| 日韩熟女一级片在线观看| 窝人体色www视频婷婷| 国产视频欧美一区二区| av网站大全在线免费观看| 91精品国产一二三产区区别在哪| 另类专区亚洲欧美| 五月婷婷高清在线观看| 亚洲 欧美 中文字幕第一页| 精品伊人久久免费视频| 天天天操天天干天天日| 极品在线激情av| 蜜桃福利视频一区二区| 绯色av一区二区三区在线观看| av在线一区二区三区地区| 制服丝袜美腿一区二区在线观看| 日韩午夜福利视频合集| 亚洲一区二区精品无码99| 91在线视频播放地址| 少妇精品久久综合网| 青青草原精品视频在线观看| 亚洲综合色区另类小说| 久久久精品伦理一区二区三区| 超碰人妻人人爱人人干| 日韩深喉口爆吞精| 91国语对白精品露脸| 久久国语露脸精品国产麻豆| 日韩欧美特级一级二级| 国产91熟女高潮一区二区| av男人的东京天堂热| 五十路熟女人妻在线网观看| 久久国产精品免费看三级| 极品少妇激情av在线| 最近最好的中文字幕免费| 18禁美女国产美女网站| 精品91久久99九九| 久久人妻精品系列蜜桃| 出轨少妇自白小说| 中文字幕一区二区三区四区二区| 变态另类影音资源| 欧美日韩性视频播放| 婷婷综合亚洲精品久久| 人妻一区两区三区四区| av伊人网好吊妞| 五月婷婷色激情网| 日韩欧美一区黄色| 亚洲一区二区三区三州| 伊人网在线播放av电影| 一道久久久综合狠狠| 久久一本色道综合六月天| av不卡中文字幕在线观看| 精品人妻熟女一区| 日韩欧美亚洲精品人妻| 99热精品夜夜爽伊人| 日韩免费视频播放| 亚洲成人激情自拍| 天堂性视频99在线播放免费| 欧美日韩操操操操操| 午夜蜜臀一区二区三区四区五区 | 婷婷综合五月激情网| 91在线视频亚洲| 五月婷婷色激情网| 国产精品一级999| 中文在线字幕丝袜美腿| 日韩人妻中文字幕视频| 亚洲av男人的天堂久久精品| 玖玖在线视频精品| 亚洲av日韩精品久久国产多毛女| 亚洲熟女少妇一区二区三区在线| 激情欧美一区二区三区| 亚洲av综合av一区二区综| 国产一区精品视频免费播放| 丁香六月天久久婷婷| 久久九精品综合丝袜影视精品| 18成人黄色在线观看| 在线 人妻 视频| 免费av资源网址| 亚洲 制服 中文字幕| 人人妻人人澡人人爽人人精品直播| 久久精品国产亚洲av麻豆看片| 99热6在线观看高清完整版| 精品人妻三区日日| 亚洲精品男人黄色的天堂| 丁香婷婷综合久久来来去| 国产自拍在线网站| 亚洲黄页在线视频| 亚洲精品又黄又爽的视频| 在线中文字幕国产精品| 中文字幕日韩人妻视频一区| 国产精品99久久久久久有的能看| 亚洲五月六月激情| 色av色婷婷18人妻久久久| 日韩毛片成人在线| 国产精品黑丝美腿美臀| 日本免费观看一区| 亚洲另类综合小说| 2019中文字幕免费在线观看| 国产91在线播放网址| 丝袜美腿诱惑福利| 在线播放一区日韩| 久久久久日本精品少妇| 高清不卡二卡三卡四卡无卡| 日韩毛片av网站| 国产精品欧美久久久无广告| 日韩精品视频在线视频看看| 2020国内精品免费视频| 国产精品久久国产精品99-| 蜜臀国产综合久久第一页| 午夜免费观看国产视频| 欧美日韩久久理论视频| 高潮喷水在线欧美| 中文字幕在线视频在线| 国产精品视频一区二区三区16| 色婷婷久久综合久色 | 熟女阿b老熟女一区| 成人国产av免费视频下载 | 日韩色图欧美视频| 视频在线观看色版| 欧美中文字幕精品在线观看| av网站大全在线免费观看| 亚洲国产婷婷久久久久久一区| 激情内射一区二区三区| 免费看的日本一级片| 91人妻人人澡人人爽人人精品6| 日本精品久久久久电影网| 91精品伊人久久久大香线蕉91| 亚洲444kkkk在线观看| 亚洲中文av天堂| 国内精品久久久久久精品视频| 黄色成年人在线观看| 清纯唯美亚洲五月激情综合| 欧美日韩视频高清一区二区 | 亚洲国产a∨天堂| 日韩美女精品视频| 日本妇女高潮视频| 欧美一区二区三区激情久久| 91熟女国产老熟女| 97在线观看国产免费视频| 中文字幕中文字幕中文字幕| 亚洲日本熟妇高清| 亚洲欧美精品卡一卡二卡三| 亚洲情色一区在线观看| 超碰97免费公开在线人妻| 国产激情啪啪小视频| 久久婷婷视频一二三区| 色婷婷亚洲午夜激情| 婷婷最新精品在线| 瑟瑟视频免费观看在线| 国产精品999动漫| 99热只有国产在线精品| 熟女乱一区二区三区四区| 国产精品久久久久久久久av| 免费看黄色的网站91| 97人人模人人爽人人喊38| 91一区二区三区在线视频| 日韩两性视频在线观看| 精品久久国产精品久久| 色综合久综合久久综合久鬼88 | 在线亚洲无av码| 人妻 丝袜美腿 中文字幕| 亚洲av日韩精品久久国产多毛女| 日韩精品少妇熟女| 日韩伦理在线观看一区视频| 蜜桃av在线观看一区| 丁香婷婷综合久久来来去| 欧美二区三区精品在线| 2021狠狠狠狠狠| 亚洲av少妇高潮150p| 91热视频在线观看| 日韩中文字幕一区二区高清| 日韩在线观看午夜精品福利| 青青操视频在线观看免费观看| 婷婷丁香亚洲五月天| 国产精品熟女高潮久久99| 国产欧美日韩精品一区二区| 国产自拍免费在线观看视频| 久久网99精品国产亚洲av| 亚洲99福利视频| 亚洲av综合av一区二区综| 人妻熟妇视频专区| 国产欧美成人精品久久| 在线视频青青青草| 制服丝袜亚洲另类| 精品人妻一区二区三区四在五区| 麻豆av熟女少妇精品| 最新欧美激情一区二区| 日韩欧美国产12区| 日韩 视频一区视频 二区| 国产欧美日韩在线观看免费| 中文字幕人妻少妇一区二区| 日本中文字幕诱惑| 亚洲国产精品av久久久软件| 免费观看日韩黄色大片| av网站国产在线| 亚洲综合精品伊人久久| 日本东京热狠狠干视频| 久久精品av网站在线播放| 蜜桃久久久久久欧美| 97色伦午夜国产亚洲精品| 久久日韩精品中文字幕人妻熟女| 久久久中文字幕人妻| 大奶子av在线播放| 国产情侣自拍成人| 国产精品岛国久久久久久| 97人妻免费视频精品| 情趣视频在线国产| 天天爽夜夜爽人妻va| 亚洲av综合av一区二区综| 黄色污污污免费在线观看网站| 欧美专区日韩精品一区二区| 国在线产视频91高清| 婷婷国产人妻一区二区三区| 97狠狠狠狠狠狠狠| 超碰在线观看97视频| 欧美国产精品久久九九| 中文人妻少妇爽爽视频| 欧美二区三区精品在线| 一本之道久久久久久久| 天天日天天舔y透| av中文字幕在线观看中文| 久久精品国产99久久久露| 美女让男人捅尿眼捅爽| 91精品视频在线观看免费版| 大香蕉伊人久久草| 国产亚洲精品久久久一区二区| av中文字幕在线观看中文| 在线人妻视频观看| 久久九九在线观看视频三级| 久久精品视频在线看15| 国产老妇伦国产熟女老妇视频| 国产精品人妻系列23p| 日本伊人网在线播放| 国产99精品视频免费观看| 黑人巨大欧美一区二区视频| 国产av 一区二区三区| 国产中文在线二区三区免| 久久久久久一级二级三级| 日本冢本中文字幕| 亚洲黄页在线视频| 五十路六十路素人熟女| 国产一区二区av少妇| 国产一区二区欧美视频| 男人能不能亲女生的秘密| 少妇人妻在线一区二区| 亚洲男人av天堂精| 大香蕉伊人久久草| 天天影视色最新色| 国产又大又黄又粗在线观看| 婷婷最新精品在线| ysl蜜桃色7v| 亚洲av日韩精品久久国产多毛女| 小草青青手机视频| 天天摸天天摸天天摸| 老男人与老熟女的日逼逼| 国产99精品久久久久久久| 日韩一区二区在线观看视频| 精品高清亚洲国产欧美| 人妻视频在线免费播放| 天天日天天干天天插天天爱天天射| 午夜精彩视频免费观看| 俺来也官网欧美久久精品| 99国产精品人妻无码一区二区| 国产九色91在线视频| 99日韩欧美在线观看| 成人伊人精品色xxxx视频| 日韩午夜福利视频合集| 91高清视频在线一区| 成人伊人精品色xxxx视频 | 国产黑色丝袜在线| 激情综合婷婷久久啪啪| 亚洲中文资源在线| 啪啪啪啪啪啪啪啪啪啪网址| 亚洲一区日韩三级| 国产精品v欧美精品v日韩| 尤物av在线播放| 日本亚洲欧洲色a| 在线视频国产香蕉岛国| 青青青青啪啪啪啪网站| 熟女人妻中文字幕一三区| 精品丰满少妇人妻| 久久亚洲天堂成人精品| 97精品人妻一区蜜桃| 日韩人体高清精品一区二区三区| 成人动漫在线观看播放| 尤物av在线播放| 欧美性猛交xxxx免费看久久| 日韩 视频一区视频 二区| 国产精品人妻一区二区三区| 日韩深喉口爆吞精| 99熟女精品一区二区三区 | 九九九九九九久久久久久久伊人| 亚洲精品人成网址| 亚洲一区两区三区四区| 亚洲av不卡福利| 久久精品视频在线看15| 国产精品三级在线免费观看 | 高清视频区二区三区| 色综合久综合久久综合久鬼88 | 国产一区二区精美视频| 日韩av另类在线| 久久国产午夜精品| 久草视频在线有精品| 国产69精品久久久久久久a| 老司机中文视频网| 日韩美女中文字幕在线看| 综合九九久久九九久久| 男女啪啪高清网站| 黑人巨大欧美一区二区视频| 偷拍 自拍 在线| 日韩啊v视频在线| 曰曰摸夜夜添av老司机蜜桃视频| 一区二区三区在线91| 麻豆av熟女少妇精品| 青青操视频在线观看免费观看| 亚洲综合日韩另类| 亚洲av操伊人久久| 91一区二区久久国产乱| 蜜桃福利视频一区二区| 亚洲欧美精彩视频| 中文字幕人妻久久不卡| 免费看黄色的网站91| 中文字幕 日韩 麻豆| 精品伊人久久免费视频| 国产精品国产三级av| 自拍偷拍国产在线| 成年人看免费黄色片| 久久凹凸视频在线观看| 精品欧美日韩免费| 日韩欧美一区二区在线免费观看| 国内偷拍视频久久久久久| 99熟女精品一区二区三区| 老熟妇淫老妇女av| 久久人妻大香蕉艺儿网| 天天日天天干天天插天天爱天天射| 亚洲视频精品一区二区三区四区| 欧美裸体视频一区二区三区| 九九在线观看视频国产剧情| 日本黄色日比视频| 在线欧美日韩观看一区二区| 91精品老司机视频网站| 中文字幕无线码在线观看| 亚洲av少妇高潮150p| 天天操美女综合网| 伦理片一区二区三区在线观看| 亚洲av一区二区色| 亚洲精品在线免费观看av| txtv在线视频| 欧美经典一区二区三区| 人妻 在线 二区| www.亚洲综合色| 国产一区二区三区女人的床叫声| 黑人巨大欧美一区二区视频| 91高清视频在线一区| 国产精品欧美日韩精品| 日韩欧美卡一卡二卡三| 日韩少妇一区二区三区四区| 天天天操天天干天天日| 熟女探花88av| www.狠狠视频.com| 经典国产91精品福利网站在线看| 亚洲av不卡福利| 久久97精品久久久久久久不卡| 欧美精品国产字幕| 国产又黑又粗又黄又大| 九色原创自拍视频| 亚洲人成电影aaa| 青青操视频在线观看免费观看| 亚洲亚洲亚洲天堂天堂天堂| 自拍另类亚洲欧美| 美女人妻素人在线丝袜| www.日产国产欧美com| 日本熟妇肉体裸交| 日本高清久久久久久久| 不卡的av中文字幕在线观看| 男人天堂手机在线2014| 久久精品国产亚洲av麻豆艾秋| 三级有码在线观看| 色婷婷亚洲午夜激情| av中文字幕在线观看中文| av一区二区三区久久久| 大尺度做爰啪啪床戏欧美| 久久女人撒尿视频| 色婷婷五月亚洲大全| 欧美成人亚洲另类图片小说网| 2020午夜免费福利| 亚洲中文字幕精品久久app| 在线成人激情av| 黄色成年人在线观看| 精品久久久久久五月天| 97人妻免费视频精品| 欧美成人金8天国加勒比| 亚洲三级av一区| 亚洲乱码久久中文| 日本伦理在线不卡| 黄色午夜免费网站| 97特黄一级中文字幕在线| 久久久久久久久久久久久福利| 亚洲av影视综合| 成人伊人精品色xxxx视频| 亚洲狠狠插2020| 男人添女人逼免费全视频| 久久久精品伦理一区二区三区| 日韩欧美特级一级二级| 日韩精品在线午夜| 亚洲国产天堂资源| 综合九九久久九九久久| 亚洲蜜桃av妇女| 亚洲青青青草在线免费视频| 人妻人伦精品国产| 亚洲av成人一区午夜网站| 国产又粗又猛又黄又爽的视频| 伊人网综合高清在线播放| 亚洲精品亚洲成人| 国产精品一区二区流白浆| 黑人巨大欧美一区二区视频| 国产熟女av一区| 国产高跟丝袜av| 久久久久亚洲国产av| 999久久久精品视频在线观看| 日韩三级中文字幕在线播放| 久久久久久91亚洲精品| 亚洲视频成人在线播放| 亚洲 欧美 日韩 人妻在线| 熟女 日本 在线| 开心五月婷婷丁香婷婷| 狠狠婷婷久久精品一区二区| 一区二区三区亚洲社区| 亚洲一区两区三区四区| 国产精品色悠悠在线观看| 日韩欧美一区黄色| 日韩毛片免费播放| 最新福利网址91| 人妻一区二区三区精品高| 91精品夜夜夜一区二区三区老板| 欧美一区二区三区夫妻| 使劲点爽视频网站| 1024你懂的日韩欧美人妻| 精品国产免费久久久久尖叫| 美女网站视频大全| 狠狠综合久久综合88亚洲| 夜夜嗨av老熟女一区二区三区| 成人日本免费视频| 精品人妻交换视频在线看| 亚洲av狠狠做五月| 亚洲av日韩美av在线播放| 91影院免费破解污在线观看| avove在线视频| 天天天天天天天天天天天天天天干| 国产福利在线观看91| 欧美一级二级三级久久| 蜜桃av一区二区精品| 超碰资源免费在线| 日本五十路人妻hd| 亚洲色影在线视频| 久久国语露脸精品国产麻豆| 国产精品一级999| 天天色天天做天天爽| 玖玖在线视频精品| 久久综合日日夜夜| 色综合久综合久久综合久鬼88| 亚洲av一区二区色| 激情亚洲精品在线观看| 欧美精品视频一二三| 超碰人人97青青草| 麻豆日韩在线视频| 中文字幕亚洲素人| 一区二区三区亚洲社区| 久久久久久激情av性色| youtube没有中文字幕| 久久久99久久久蜜桃| 日本中文字幕电影在线观看 | 日本女人干逼视频| 视频一区视频二区中文精品| 日韩伦理免费大片| 午夜免费福利视频| 亚洲国产一区91| 奶头被吸得又大又黑np| 欧美熟妇久久久久久久久久久| 亚洲岛国av在线| 少妇人妻一区二区| 男人添女人逼免费全视频| av国产网站在线观看| 亚洲国产久一区二区三区| 曰曰摸夜夜添av老司机蜜桃视频| 99免费国产精品视频| 国产老妇伦国产熟女老妇视频| 情趣视频在线国产| 久草视频播放在线| 国语自产拍在线观看视频| 日韩欧美a级视频在线免费观看 | 乱码欧美中文字幕日韩| 蜜臀久久午夜小视频,| 国产精品久久久久噜噜噜| 国产精品视频福利在线| 亚洲伊人久久在线| 99久re热视频这只有精品6| 国产精品三级在线免费观看 | 免费黄色特级大片| 女人的天堂av网| 97电视剧在线观看免费| 黄色午夜免费网站| 中文字幕av人妻呻吟| 国内美女直播视频| 亚洲一区二区精品无码99| 亚洲中文字幕乱码七糟| 欧美性受xxxx人妻xyv狂| 久久视频这里只精品| 人妻精品一二三区| 福利视频一区二区三区| 999热视频精品在线| 亚洲国产精品va在线观看www | 日韩毛片av网站| 蜜桃视频一区二区免费| 亚洲成人天堂久久| 大香蕉av在线免费观看| 色婷婷欧美亚洲激情综合另类在线| 巨乳中文字幕一区| 国产夫妻自拍啪啪视频| 亚洲色影在线视频| 萌白的所有视频在线观看| 超碰97在线在线观看| 久久午夜一成年人的视频| 喝醉漂亮人妻被强了中字| 欧美视频亚洲视频在线观看| 青青草原国产在线精品| 婷婷国产成人在线| 天天操天天透天天射| 九九热九九色九九操| 中文字幕久久j日| 欧美国产日本精品| 亚洲成人av电影播放| 日韩熟女在线视频| 国产高跟丝袜av| 少妇人妻一区二区| 午夜精品老牛av一区二区三区| 日本大香蕉怡红院| 国产乱品免费一区二区三区视频| 成人在线电影一区二区| 中文精品久久久久人妻不| 少妇熟女一二三区| 亚洲男人av天堂精| 天堂av中文字幕乱码免费看| 国产99精品视频免费观看| 99热这里只有精品在线| 日韩中文av在线观看| 人妻熟妇视频专区| 黄黄的视频靠在线观看| 97国产一区二区三区| 国产成人午夜三级在线观看| 成人国产精品一区二区视频下载| 色视频在线一区二区三区| 人妻丰满熟妇av一区二区| 日韩中文乱码字幕| 国产婷婷精品av在线| 亚洲444kkkk在线观看| 日本高清中文字幕在线| 中文字幕天堂av.| 暖暖视频高清在线观看中文| 深夜美女福利诱惑| 久久久久久精品国产婷婷| 香蕉久久成人国产精品| 一区二区人妻字幕| 成人精品永久免费观看| 精品毛片av一区二区三区| 国产日韩精品自拍在线| 最新人妻激情视频网| 中国精品久久久久久| 日韩国产主播av在线| 在线播放一区日韩| 亚洲国产日韩欧美精品综合| 日韩精品中文字幕熟女少妇| 日韩欧美在线观看二区| 麻豆成人性电影在线观看视频| 国产精品日韩欧美婷婷综合久久| 中文字幕久久j日| 亚洲av不卡福利| ff14一区二区三区分别是啥| 久久午夜激情视频| 中文字幕亚洲自拍偷拍| 日韩国产欧美激情在线视频| 伊人久久精品在线观看| 欧美一区三区三区高中清不卡| 中文字幕精品久久久| 91精品久久久老熟女9久| 欧美视频亚洲视频在线观看| 久久av一区二区三区neco| 国产亚洲成人av看黄在线观看| 天天色天天干天天| 63久久av精品一区二区| 青青青在线视频人视频| 国产99精品久久久久久久| 国产精品亚洲综合一区在线观看| 蜜桃福利视频一区二区| 国产在精品在线观看| 三级日本黄页网站大全| 久久久久大香青草精品综合| 欧美专区日韩精品一区二区| 97一区二区三区人妻免费| 新老鸭窝在线视频| 麻豆国产成人在线视频网站| 日韩在线成人中文字幕| 激情五月婷婷婷婷婷婷婷| 久久亚洲中文字幕精品| 邻居天天cao我1v1高| 国产日韩高清一区二区| 亚洲69偷拍视频在线观看| 亚洲中文自拍偷拍| av少妇人妻中文字幕| 国产av熟女一区二区三区四季 | 欧美日韩国产最新久久| 亚洲av懂色av| 亚洲男人的天堂久久| 中文字幕人妻熟女一区二区三区| 久久久久久久美女特黄大片| 国产黄色一级黄色大片| 中文一区二区人妻| 夫妻性生活一级黄色大片| 九九热九九色九九操| 亚洲天堂黄色网络| 中文字幕日韩精品在线免费| 91青青草这里只有精品| 亚洲春色男人天堂| 婷婷久久综合久色综| 九色视频蝌蚪在线免费观看| 中文字幕人妻一区二区三区熟女| 国产蜜桃av在线观看| 男女啪啪高清网站| www.狠狠视频.com| 日韩av手机免费在线观看| 成人黄色在线电影网站| 99熟女精品一区二区三区| 亚洲av极品在线| 国语自产拍在线观看视频| 国产欧美日韩精品一区二区| 自拍视频在线观看一区| 久久亚洲国产精品电影| 夜夜嗨一区二区三区直播平台| 黄色一级片人和兽| 在线97视频观看| 成人鲁啊鲁是男人的天堂| 色婷婷综合久久久精品视频| 熟女人妻久久中文字幕| 超碰97免费公开在线人妻| 亚洲欧美精品卡一卡二卡三| 青青操操操操操操操 | 国产精品偷伦免费视频| 91九色porn在线| 国产精品综合久久桃花| aise美乳诱惑| 国产一级aa特黄大片| 久久久久久久精品欧美| 欧美国产日本精品| 色婷婷激婷婷深爱五月老司机| 中文字幕 一区二区| 日韩av在线播放中文字幕| 天天插天天日天天啪| 麻豆剧传媒精品国产av| 精品熟女少妇av久久图| 1024在线国产视频| 亚洲69偷拍视频在线观看| www.99久久久久.com| 国产亚洲欧美激情片| 日韩人妻精品一二三| 自拍偷拍另类图区| 99久久精品国产亚洲av热热爱| 有没有黄色一级片| 天天色天天干天天| 丰满人妻一区二区免费视频| 国产精品v欧美精品v日韩| 九九热免费视频播放| 偷拍99免费视频| 40路熟女人妻aⅴ一区二区三区| 中文字幕在线三级| 国产情趣丝袜高跟av色| 亚洲综合视频麻豆| 有没有黄色一级片| 北条麻妃人妻在线| 在线播放一区日韩| 中文字幕日韩精品在线免费| 亚洲欧美日韩综合在线丁香| 中文字幕一区在线精品| 日韩三级 国产精品| 国产亚洲精品美女久久久| 欧美国产日本精品| yy8098国产av久久| 激情亚洲精品在线观看| 亚洲国产一区91| 超碰97免费公开在线人妻| eeuss一区二区人妻| 99久久成人精品国产网站| 国产一区区二区三区| 一色桃子av人妻中文字幕| 欧美精品中文字幕久久久久| 亚洲精品av网站| 18成人黄色在线观看| 国产 91 九色 精选| 精品人妻av区乱码久久密臀| 国产精品99久久久久久有的能看| 午夜在线视频播放网站| 91一区二区三区在线视频| 欧美视频在线播放一卡| 久久少妇高潮视频免费| 亚洲av久久精品在线| 日本免费久久精品| 日韩精品亚洲性图欧美| 大香蕉大香蕉婷婷丝袜大香蕉| 久久 91 欧美 精品| 国产免费播放一区| 成人午夜免费在线视频网站| 免费一区二区风骚徐娘| 日韩欧美一卡二卡在线观看视频| 久久国产成人精品免费视频| 欧美成人破处视频| 久久一区二区三区av| 亚洲一区二区精品无码99| 少妇熟女一二三区| 成人在线电影一区二区| 日韩成人免费电影一区二区三区 | 无套内射处女在线观看 | 亚洲人妻自拍偷拍| av网站软件在线播放| 在线 人妻 视频| 亚洲熟女少妇一区二区三区在线| 亚洲综合制服丝袜另类在线| 使劲点爽视频网站| 日韩中文字幕无码中文字| 国产自拍免费在线观看视频| 在线观看国产高清精品| 久久久久久亚洲精品不卡| 亚洲经典 中文字幕| 国产网址视频在线观看| 久久 91 欧美 精品| 日韩三级中文字幕在线播放| 日韩亚洲在线成人| 国产 91 九色 精选| 88人妻x88av| 玩弄丰满人妻一区二区av| 五月亚洲婷婷亚洲| 久久成人综合亚洲精品欧美| 国产99精品久久久久久久| 久久伊人中文字幕有码| 亚洲综合视频麻豆| 亚洲最大黄色录像| 久草这里只有精品99| 爱片av在线观看| 尤物在线观看视频av| 五月天色婷婷激情小说久久综合| 骚片av蜜桃精品一区| 不卡av中文字幕手机看| 欧美亚洲成人动漫在线| 久久久久久久久久久久久福利| 大桥久未无码吹潮在线观看| 99久久综合精品五月天| 91九色porny国产| 女同性恋亚洲av| 国产麻豆丝袜美腿诱惑在线播放| 午夜影院成人av| 精品888欧美久久久| 精品91久久99九九| 欧美伊人亚洲伊人色综合动图| 日本大胆女人视频| 超碰大香蕉在线一人| 黄色强奸片免费观看视频免费看| 国产 欧美?亚洲?日韩视频| 人人妻人人澡人人爽人人a v| 成年女性午夜爽爽爽在线看片| 一区二区三区亚洲社区| 国产精品高清国产三级av| 国产一级激情黄色av| 喝醉漂亮人妻被强了中字| 2019中文字幕免费在线观看| 亚洲欧美日韩一区另类| 午夜8050网站二级| 中文人妻精品一区二区三区四区| 色婷婷五月亚洲大全| 人妻乱干视频免费| 国产又粗又长的视频| 国产 日韩 精品在线观看| 中文av字幕在线观看一区| 自拍视频一区在线| 国产精品国产三级av| 视频一区二区蜜桃| 91色综合综合热五月激情| 开心五月婷婷丁香婷婷| 伊人色综合久久久久| 蜜桃成人永久免费av大| 日韩国产欧美激情在线视频| 青青草免费国产视频| 亚洲第一蜜桃av| 欧美一区二区三区,视频| 国内自拍全部视频在线| 久久久99久久久蜜桃| 青青草原国产在线精品| 自拍偷拍国产在线| 91九色精品福利视频| 美女黄页网站久久久久久| 久久91精品国产91久久蜜月| 人人妻人人澡人人爽人人a v| 国内精品伊人久久久久av网站| 暖暖视频高清在线观看中文| 亚洲一区二区 偷拍| 美女精品福利视频在线观看| av日韩av亚洲国产| 日韩av在线大片| 欧美视频一区二区三区久| 色综合久久中文综合久久| 日韩免费av在线观看激情| 国产欧美在线亚洲| 人妻中文字幕精品| av最新在线中文天堂| 欧美情色伦理在线| 成人在线免费观看久久视频| 亚洲色影在线视频| 日韩毛片成人在线| 日本女优在线三区| 男人日女人的逼的视频| 麻豆精选国产在线av观看| 91精品久久久老熟女9久| 亚洲黄色影院首页| 麻豆成人性电影在线观看视频| 国产精品999动漫| av在线免费观看天堂| 亚洲另类伦春色综合小| 国产91精品福利在线观看| 中文字幕人妻少妇一区二区| 视频免费在线你懂的| 激情亚洲成人一区二区三区| eeuss一区二区人妻| 亚洲丝袜美女诱惑| 日本高清中文字幕在线| 日韩欧美中文字幕第一页| 变态另类天堂在线播放 | 亚洲最大黄色录像| 国产精品久久久久久人妻爽 | 人人妻人人澡人人爽人人a v| 中文字幕亚洲综合色| 自拍偷拍另类图区| 日韩不卡在线av| 久久99 国产精品| 亚洲中文av字幕综合| 黄色一级网站免费在线播放| 熟女人妻中文字幕一三区| 久久91精品国产91久久蜜月| 国产熟女啪啪精品一区二区三区 | 国内偷拍视频免费久久| 日韩国产主播av在线| 色av色婷婷18人妻久久久| 亚洲淫中文字幕一区| 青青草欧美激情在线视频| 人妻 在线 二区| 亚洲精品在线17| 17c久久精品国产亚洲| 日韩欧美卡一卡二卡三| 日韩欧美在线情色| 手机av永久免费| 亚洲精品丝袜美腿久久| 中文字幕亚洲天堂| 91国产免费视频国产免费| 国产精品美女视频免费线播放| 蜜臀国产综合久久第一页| 日本久久一区二区三区| 99久久九九社区精品| 亚洲黄色片在线观看| 大香蕉av网站在线观看| 精品人妻一区二区免费| 亚洲 制服 中文字幕| 人妻一区二区三区精品高| av一区中文字幕在线| 91捷克街头在线播放| 中文人妻熟妇精品乱又伧不卡| 亚洲av操伊人久久| 新老鸭窝在线视频| 婷婷在线www.| 尤物av在线播放| 亚洲字幕中文精品| 久久国产精品亚洲欧美阿娇| 欧美一区二区三区国产| 日韩av激情视频| 欧美熟妇大阴蒂高潮系列| 国产成人h片在线观看| 少妇被爽到高潮喷水久久| 日韩欧美一区二区在线免费观看| 亚洲国产长腿丝袜av天堂| 丰满少妇久久久久久久的| 国产一区二区精美视频| 精品人妻一区二区三区四在五区| 中文字幕久久久一区久久久青春| 俺去啦视频在线精品| 日本久久一区二区三区| 日韩亚洲欧美高清视频| 亚洲经典 中文字幕| 日韩av激情视频| 91精品又粗又硬又爽少妇的视频 | 国产成人午夜电影在线观看免费| 在线视频一区二区观看| 视频一区二区三区免费观看| 不卡一卡2卡3卡4卡精品在| 国产69精品久久久/| 高清有码中文字幕在线观看| 日本中文字幕电影在线观看| 婷婷久久中文字幕17| 手机av资源在线| 久久视频这里只精品| 国产伦精一区二区三区| 国产一区视频免费观看| 精品黄色美女一二区| 国产亚洲精品av成人无删减| 中文字幕天堂av.| 婷婷av在线免费观看| 国产又黑又粗又黄又大| 精品久久国产精品久久| 淫妇操BBB操BBB操BBB| 日韩精品中文字幕巨臀人妻中出| 91最新视频在线播放| 国产精品高清在线观看| 91九色porn在线| 丝袜人妻精品一区二区三区| 福利无码一区二区三区| 日韩一区二区在线观看视频| av人妻日本不卡二区| 久久久久久 九色av| 蜜桃精品一区二区三区免费看| 亚洲少妇av一区| 欧美日韩久久理论视频| av人妻精品一区二区三区| 色福利视频导航网| 日韩精品免费在线观看视频| 天天干天天综合色| 亚州av男人天堂| 麻豆剧传媒精品国产av| 999久久久精品视频在线观看| 青青手机国产视频| sese欧美日韩| 国内精品久久99人妻无| 日韩 欧美 国产 丝袜 | 久久精品国产露脸对白| 日本精品九九久久精品一本| 国产欧美日韩精品电影免费| 午夜精彩视频免费观看| 日本高清中文字幕在线| 亚洲国产久一区二区三区| 青青草免费国产视频| 免费黄色特级大片| 色婷婷亚洲午夜激情| 自拍伦理视频在线观看| 一道久久久综合狠狠| 少妇精品久久综合网| 97香蕉碰碰人人澡人人爱| 日韩人妻激情瑟瑟| 日本五十路六十路熟妇| 亚洲啪啪啪一区二区三区| 日韩中文av在线观看| 成人欧美三级视频| 2025av熟女| 亚洲另类综合小说| 91涩漫在线观看| 国产中文在线二区三区免| 亚洲精品女人久久| av岛国网站在线观看| 日韩av一区二区三区人妻| 亚洲av久久精品在线| 青青草免费国产视频| 成人精品国产一级二级| 国产中文在线二区三区免| 国产精品久久久久久久久av| 欧美日韩成人在线高清| 人妻 制服 日韩 中文 在线| 黑人玩弄人妻一区二一区三区| 亚洲欧美精品夜色在线网站| 日本熟妇vvw日本人妻| 4438x中文字幕| 日韩av天堂黄色片| 小明看看成人在线免费视频| 国产精品久久久久噜噜噜| 一本色道久久爱久| 欧美极品激情一区二区三区| 100xxoo蜜乳| 无码人妻丰满熟妇区五十路| 国产麻豆色哟哟网站| 日韩中文字幕在线第一页| 午夜美女免费福利| 欧美日韩精品变态另类| 邻居天天cao我1v1高| 东京热制服人妻诱惑| 认你日爽在线视频| 97人妻免费视频精品| 91国产老熟女精品| 日本五十路人妻hd| 啊啊嗯嗯好爽视频| 国产一区精品视频免费播放| av一区中文字幕在线| 日本五十路人妻hd| 蜜臀日韩一区二区三区| 久久综合日韩欧美| 国家五a级久久久久久黄片| 日韩中文字幕综合在线网| 亚洲成人激情av在线播放| 黑人巨大超大另类videos| 日韩在线观看午夜精品福利| 亚洲男人av天堂精| av天堂地址在线观看| 久久精彩视频免费看| 天天操天天射天天干天天爱| 女同性恋亚洲av| 激情av五月婷婷| 久久精品亚洲精品无码白云tv| 亚洲av中文字幕| 国产精品999动漫| 中文字幕人妻aaaaa| 中文人妻一区二区视频| 免费观看日韩一级黄色大片| 亚洲第一蜜桃av| 国产超碰人人一区二区三区| 极品人妻口爆颜射| 亚洲国产精品资源在线| 熟女 av在线 一区二区| 久久久久久久久久一二三| 国产中文在线二区三区免| 国产精品最新自拍| 天天操美女综合网| 1024你懂的日韩欧美人妻| 成人黄色av在线播放| 熟女91n一区二区三区| 国产网址视频在线观看| 视频不卡在线观看| 欧美一区二区三区综合网| av人妻日本不卡二区| 国产又粗又猛又黄又爽的视频| 国产精品免费自拍视频大全| 999久久久精品国产| 欧美日韩av一区=区三区| 91精品爽啪在线观看| 日韩av激情视频| 搜索人妻av中文字幕| 国产精品va一级二级三级| 日韩少妇一区二区三区四区| 岛国黄色在线网站| 亚洲五月婷婷啪啪| 欧美专区日韩精品一区二区| 91在线视频亚洲| 精品人妻av区乱码久久密臀| 亚洲中文字幕久久精品品| 都市激情 校园春色 中文字幕| 熟女少妇人妻中文字幕| 玖玖在线视频精品| 91精品爽啪在线观看| 人妻精品久久久久久| 在线观看国产高清精品| 蜜臀国产综合久久第一页| 精品久久久久久久久99| 香蕉久久久久久久av网站| 天天色天天干天天| 99久精品综合在线观看| 国产精品久久久久免费播放| 极品少妇高潮久久久久久久久| 全网小视频资源免费在线播放| 人妻av乱片av出轨| 亚洲色影在线视频| 日韩亚洲欧美高清视频| 中文字幕免费无卡| 久久精品久久久久久久久久| 美腿丝袜综合在线日韩| 日韩女人小便视频| 日本视频一区免费| 91亚洲精品资源| 欧美经典一区二区三区| 自拍另类亚洲欧美| 97在线观看国产免费视频| 一区二区三区四区丝袜| 国产视频97在线播放| 小明看看成人在线免费视频| 成人国产av免费视频下载| 久久伊人青青草大香蕉| 久久99 国产精品| 日韩少妇一区二区三区四区| 东京热人妻视频在线观看| 人妻精品久久久久久| 欧美日韩亚洲成人一区| 经典a级片在线观看| 欧美国产日韩美女主播视频一区| 日韩伦理不卡一区二区| 婷婷六月亚洲综合| av在线亚洲最大的| 中文字幕人妻一区二区三区熟女| 日本aaaaa级特黄大片老头| 丝袜人妻一区二区三区网站| 亚洲成人偷拍av| 黄页网站网址在线观看| 激情中文字幕视频| 亚洲一卡2卡3卡4卡5卡| 欧美裸体视频一区二区三区| 制服丝袜美腿一区二区在线观看| 亚洲黄色尻逼网站| 国产精品一一色哟哟| yeye38亚洲综合网久久| 日韩毛片短视频在线播放| 国产一区二区欧美视频| 蜜桃一区二区三区在线观看视| 91高清在线观看一区| 欧美一区二区三区,视频| 蜜桃一区二区三区免费在线观看| 国产三级国产精品久久成人| 天天色天天舔天天干| 高清不卡二卡三卡四卡无卡| 国产av专区网站大全| 天天射天天热天天舔| 观看国产精品97视频| 久草视频在线有精品| 久久综合亚洲狠狠伊人| 日韩亚洲欧美高清视频| 一道久久久综合狠狠| 疯狂人妻丝袜系列| 亚洲男人的天堂色偷免费| 成人午夜毛片在线| 大龄岳母的诱惑电影在线观看| 午夜天堂久久电影|