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

桌面掛件不能承受之重——GIF

開發
本文從桌面掛件開發過程中遇到的GIF圖片難以加載的問題展開,分別介紹了現有的掛件中加載GIF圖片的兩種可行方案——ViewFlipper和AnimatedImageDrawable,同時闡述了兩種的方案的優缺點。

本文從桌面掛件開發過程中遇到的GIF圖片難以加載的問題展開,分別介紹了現有的掛件中加載GIF圖片的兩種可行方案——ViewFlipper和AnimatedImageDrawable,同時闡述了兩種的方案的優缺點。然后針對現有方案中的痛點,結合現有方案,提出通過網絡下發GIF并通過逐幀解析得到幀圖片,再采用ViewFlipper來實現加載的方案,解決痛點中的引入資源過多導致包體增大的問題,使掛件既能不增加包體又能展示GIF。

01、背  景

眾所周知,Android原生的原子組件(AppWidget,又名桌面掛件)所能使用的View有限,僅能支持如下的:

layout(布局):

  • AdapterViewFlipper
  • FrameLayout
  • GridLayout
  • GridView
  • LinearLayout
  • ListView
  • RelativeLayout
  • StackView
  • ViewFlipper

widgets(小部件):

  • AnalogClock
  • Button
  • Chronometer
  • ImageButton
  • ImageView
  • ProgressBar
  • TextClock
  • TextView

API 31開始,還支持如下的小部件和布局:

  • CheckBox
  • RadioButton
  • RadioGroup
  • Switch

需要注意一點,除了上述這些之外,其余所有的都不支持,包括繼承自這些類的子類同樣也不支持。因此我們能夠看出,開發AppWidget的局限性比較大,只有限定的布局和小部件能夠使用,且不能通過繼承來實現自定義的炫酷效果。這里也解釋了為什么筆者一開始不直接使用Lottie、PAG等來實現復雜的動畫,完全是被限制了。

不僅如此,組件內由于使用的都是Remoteviews,Remoteviews可以在其它進程中進行顯示,我們可以跨進程更新它的界面。Remoteviews在Android中的主要應用是通知欄和桌面掛件。也正式掛件中使用的是Remoteviews,所以我們不能像普通Android應用一樣使用findViewById或者viewbinding來獲取View的對象并通過view對象來設置相應的屬性等。在掛件中只能使用Remoteviews中的一些方法,這些方法基本都是通過反射方式進行封裝來實現的,比如設置ImageView的圖片,Remoteviews中只提供了如下四種方法

/**
 * Equivalent to calling {@link ImageView#setImageResource(int)}
 *
 * @param viewId The id of the view whose drawable should change
 * @param srcId The new resource id for the drawable
 */
publicvoidsetImageViewResource(@IdResint viewId, @DrawableResint srcId){
    setInt(viewId, "setImageResource", srcId);
}
  
/**
 * Equivalent to calling {@link ImageView#setImageURI(Uri)}
 *
 * @param viewId The id of the view whose drawable should change
 * @param uri The Uri for the image
 */
publicvoidsetImageViewUri(@IdResint viewId, Uri uri){
    setUri(viewId, "setImageURI", uri);
}
  
/**
 * Equivalent to calling {@link ImageView#setImageBitmap(Bitmap)}
 *
 * @param viewId The id of the view whose bitmap should change
 * @param bitmap The new Bitmap for the drawable
 */
publicvoidsetImageViewBitmap(@IdResint viewId, Bitmap bitmap){
    setBitmap(viewId, "setImageBitmap", bitmap);
}
  
/**
 * Equivalent to calling {@link ImageView#setImageIcon(Icon)}
 *
 * @param viewId The id of the view whose bitmap should change
 * @param icon The new Icon for the ImageView
 */
publicvoidsetImageViewIcon(@IdResint viewId, Icon icon){
    setIcon(viewId, "setImageIcon", icon);
}

從源碼中可以看到,setImageViewResource 方法只能傳入int類型的資源,也就是在資源文件中的資源ID,除此之外就是Bitmap、Uri和Icon類型,無法支持Drawable等類型。由此可見,組件中的View其實只能包含普通View的一部分功能,限制比較明顯。

02、掛件加載 GIF 的可行方案

言歸正傳,首先,我們介紹下在組件中加載GIF的可行方案,主要有兩種:

2.1 方案一:使用ViewFlipper來實現逐幀動畫的效果

此方案是利用Remoteviews支持的ViewFlipper控件,配合多個ImageView來循環顯示,達到類似逐幀動畫的效果。布局內容如下:

<ViewFlipper
    android:layout_width="40dp"
    android:layout_height="40dp"
    android:layout_gravity="end|center_vertical"
    android:autoStart="true"
    android:flipInterval="90">
  
    <ImageView
        android:layout_width="40dp"
        android:layout_height="40dp"
        android:src="@drawable/before_sign_in_anim0" />
  
    <ImageView
        android:layout_width="40dp"
        android:layout_height="40dp"
        android:src="@drawable/before_sign_in_anim15" />
  
    <ImageView
        android:layout_width="40dp"
        android:layout_height="40dp"
        android:src="@drawable/before_sign_in_anim28" />
  
    <ImageView
        android:layout_width="40dp"
        android:layout_height="40dp"
        android:src="@drawable/before_sign_in_anim43" />
  
    <ImageView
        android:layout_width="40dp"
        android:layout_height="40dp"
        android:src="@drawable/before_sign_in_anim57" />
  
    <ImageView
        android:layout_width="40dp"
        android:layout_height="40dp"
        android:src="@drawable/before_sign_in_anim71" />
  
    <ImageView
        android:layout_width="40dp"
        android:layout_height="40dp"
        android:src="@drawable/before_sign_in_anim85" />
  
    <ImageView
        android:layout_width="40dp"
        android:layout_height="40dp"
        android:src="@drawable/before_sign_in_anim100" />
</ViewFlipper>

ViewFlipper中的一些常用方法如下:

  • setInAnimation:設置View或ImageView進入屏幕時使用的動畫
  • setOutAnimation:設置View或ImageView退出屏幕時使用的動畫
  • showNext:調用該方法來顯示ViewFlipper里的下一個View或ImageView
  • showPrevious:調用該方法來顯示ViewFlipper的上一個View或ImageView
  • setFilpInterval:設置View或ImageView之間切換的時間間隔
  • startFlipping:使用上面設置的時間間隔來開始切換所有的View或ImageView,切換會循環進行
  • stopFlipping:停止View或ImageView切換
  • isAutoStart:是否自動開始播放

在作為動畫設置時,需要在xml文件中設置autoStart屬性為true,保證動畫能夠自動播放。

優點:

  • 各版本兼容性好,ViewFlipper是API 11時引入的,目前應該不會有比這低的了;

缺點:

  • ImageView過多,代碼也多,修改替換麻煩;
  • 在Remoteviews中,ViewFlipper的很多方法無法使用,比如停止播放等。

2.2 方案二:使用AnimatedImageDrawable來顯示GIF動畫

Android 9.0 中引入了一個新的Drawable來顯示GIF圖片:AnimatedImageDrawable,對應的xml標簽是<animated-image>,這樣一來,我們可以直接將一個GIF圖片before_sign_in.gif放到drawable目錄中,然后新建一個before_sign_in_anim.xml來引用:

<?xml versinotallow="1.0" encoding="utf-8"?>
<animated-image xmlns:android="http://schemas.android.com/apk/res/android"
    android:autoStart="true"
    android:autoMirrored="true"
    android:src="@drawable/ic_test_gif" />

其中的ic_test_gif就是我們的.gif文件。

我們可以看下AnimatedImageDrawable的屬性:

<!-- Drawable used to draw animated images(gif). -->
    <declare-styleable name="AnimatedImageDrawable">
        <!-- Identifier of the image file. This attribute is mandatory.
             It must be an image file with multiple frames, e.g. gif or webp -->
        <attr name="src" />
        <!-- Indicates if the drawable needs to be mirrored when its layout direction is
             RTL(right-to-left). -->
        <attr name="autoMirrored" />
        <!-- Replace the loop count in the encoded data. A repeat count of 0 means that
             the animation will play once, regardless of the number of times specified
             in the encoded data. Setting this to infinite(-1) will result in the
             animation repeating as long as it is displayed(once start() is called). -->
        <attr name="repeatCount"/>
        <!-- When true, automatically start animating. The default is false, meaning
             that the animation will not start until start() is called. -->
        <attr name="autoStart" />
    </declare-styleable>

從中我們可以發現,這里可以設置repeatCount循環次數,設置為0的話表示只播放一次。

此時,我們只需要將drawable設置給ImageView即可,在Remoteviews中,考慮到版本兼容問題,我們通過如下方式設置:

remoteViews.setImageViewResource(
    R.id.abnormal_static_cat,
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) {
        R.drawable.before_sign_in_anim
    } else {
        R.drawable.before_sign_in_static
    }
)

優點:資源少,一個GIF只要一個xml,且替換簡單;

缺點:只有Android9以上的系統可以用。

2.3 現有方案的痛點

上述描述的兩種方案中,都會引入很多資源文件,這必然會增加應用的包體,導致包體增大不少,因此可以考慮通過服務端下發的方式來實現,那么問題就來了:

1)、如果通過方案一,那么客戶端必須寫定一個xml,寫定一定數量的ImageView來供下發的圖片加載,當然了,可以動態的添加,但這里是組件,動態添加會比普通的view動態添加稍微麻煩些,這個我們后面再說。

2)、如果通過方案二,那么就有問題了,前面已經提到了,組件里面的ImageView是不支持通過Drawable對象來設置內容的,這就導致了就算我們能夠得到AnimatedImageDrawable對象,我們也沒辦法設置,況且要得到這樣一個Drawable,也比較困難(沒有深究如何得到)。

戛然而止了,兩個方案實現起來聽著都不太靠譜,那么有沒有什么好的方案呢?

03、可行方案探索

3.1 初探

想到這里,大家可能會問,為什么不使用Glide呢?這個強大的圖片加載庫總不會沒有這樣的方法吧?

確實,Glide給AppWidget提供了專門的圖片加載方式,其實現方式如下:

val appwidgetTarget = AppWidgetTarget(context, R.id.abnormal_static_cat, remoteViews, ComponentName(context, TestWidgetProvider::class.java))
Glide.with(context)
   .asBitmap()
   .load(url)
   .into(appwidgetTarget)

但是從上面可以看出,這個只能加載Bitmap,如果是asGif,則在into時沒有target這個選項,只能into(ImageView)。因此這個方法也行不通。

3.2 思索與嘗試

這里還要說一點,如果是將圖片下載到手機本地,再去讀取本地文件,還需要考慮存儲權限的問題,而這里是原子組件,如果需要請求權限,那么就得找一個落地頁去承載,且組件的卡片上最好也需要有這個說明,這樣的話UI改動會比較大,且如果沒有同意權限就會出現展示不了圖片的情況,這也很不友好。

綜上,只能在請求網絡圖片時就把GIF加載出來,這樣既不需要上述的那些繁瑣的權限授予過程,也不會增加包體的大小。

受到上面第一個方案的啟發,我們可以把GIF圖的逐幀圖片取出來,然后通過方案一來展示,這樣就能實現了。

3.2.1 獲取網絡 GIF 圖片

首先是拿到網絡的GIF圖片,這里我們采用Glide來獲取(Glide還是好用啊),采用Glide還有一個好處是,Glide會針對圖片作緩存,這樣我們重復加載同一張圖不會重復消耗流量:

Glide.with(context)
    .asGif()
    .load(url)
    .diskCacheStrategy(DiskCacheStrategy.ALL)
    .submit(432, 432)
    .get()

3.2.2 得到 GIF 的逐幀圖片

然后是將得到的GIF進行解析,得到逐幀的圖片,這里我們引入一個工具庫:implementation("pl.droidsonroids.gif:android-gif-drawable:1.2.24"),該庫在Vhub上已有上傳,可以直接使用:

@WorkerThread
fun getAllFrameBitmapByUrl(context: Context, url: String): MutableList<Bitmap> {
    val frameBitmaps: MutableList<Bitmap> = ArrayList()
    var gifDrawable: GifDrawable? = null
    try {
        val gif = Glide.with(context)
            .asGif()
            .load(url)
            .diskCacheStrategy(DiskCacheStrategy.ALL)
            .submit(432, 432)
            .get()
        gifDrawable = GifDrawable(gif.buffer)
        val totalCount = gifDrawable.numberOfFrames
        for(i in 0 until totalCount){
            frameBitmaps.add(gifDrawable.seekToFrameAndGet(i))
        }
    } catch (t: Throwable) {
        VLog.e(TAG, "getAllFrameBitmapByUrl Error.", t)
    } finally {
        gifDrawable?.stop()
    }
    return frameBitmaps
}

這樣我們就得到了包含GIF所有幀圖片的列表了(美滋滋~),接下來就可以根據方案一處理每一幀的圖片了。

3.2.3 加載

然后,就報錯了,lang.IllegalArgumentException: RemoteViews for widget update exceeds maximum bitmap memory usage (used: 236588800, max: 15396480)。由于Remoteviews是跨進程的傳輸,并不是傳統意義上的view,其內部是通過Binder來實現的,因此當ImageView去setImageBitmap的時候,需要注意設置進去的bitmap是否超過了大小限制。

最大的Size公式為:The total Bitmap memory used by the RemoteViews object cannot exceed that required to fill the screen 1.5 times, ie. (screen width x screen height x 4 x 1.5) bytes.也就是RemoteViews 對象使用的總 Bitmap 內存不能超過填滿屏幕 1.5 倍所需的內存,即 (屏幕寬度 x 屏幕高度 x 4 x 1.5) 字節。這個在AppWidgetServiceImpl.java中有相應的定義:

privatevoidcomputeMaximumWidgetBitmapMemory(){
    Display display = mContext.getDisplayNoVerify();
    Point size = new Point();
    display.getRealSize(size);
    // Cap memory usage at 1.5 times the size of the display
    // 1.5 * 4 bytes/pixel * w * h ==> 6 * w * h
    mMaxWidgetBitmapMemory = 6 * size.x * size.y;
}

而且,RemoteViews源碼內部維護了一個:BitmapCache mBitmapCache, 每次設置bitmap進來,都會被緩存起來,最終計算RemoteViews占用內存大小的話,也會把這塊算進去。

/**
     * Call a method taking one Bitmap on a view in the layout for this RemoteViews.
     * @more
     * <p class="note">The bitmap will be flattened into the parcel if this object is
     * sent across processes, so it may end up using a lot of memory, and may be fairly slow.</p>
     *
     * @param viewId The id of the view on which to call the method.
     * @param methodName The name of the method to call.
     * @param value The value to pass to the method.
     */
    publicvoidsetBitmap(@IdResint viewId, String methodName, Bitmap value){
        addAction(new BitmapReflectionAction(viewId, methodName, value));
    }
  
...
  
    BitmapReflectionAction(@IdResint viewId, String methodName, Bitmap bitmap) {
       this.bitmap = bitmap;
       this.viewId = viewId;
       this.methodName = methodName;
       bitmapId = mBitmapCache.getBitmapId(bitmap);
    }
  
...
  
    publicintgetBitmapId(Bitmap b){
        if (b == null) {
            return -1;
        } else {
            int hash = b.hashCode();
            int hashId = mBitmapHashes.get(hash, -1);
            if (hashId != -1) {
                return hashId;
            } else {
                if (b.isMutable()) {
                    b = b.asShared();
                }
                mBitmaps.add(b);
                mBitmapHashes.put(mBitmaps.size() - 1, hash);
                mBitmapMemory = -1;
                return (mBitmaps.size() - 1);
           }
       }
   }

這里由于GIF解析出來的幀圖片太多,如果每一張都設置的話,確實太多了,那么就需要采取采樣的方式,目前設定的是每5張中取一張,然后設置了每一張圖片的大小也不能超過閾值,另外總體也設置了一個閾值,防止超過報錯。這里就會出現兩個問題,一個是單張圖片限制了大小閾值,必定會出現壓縮、采樣,導致單張圖片質量下降,不像原先那么高清,第二個是幀圖片太多,就算單張限制了閾值,總體也會超過總體的閾值,在超過總體前一幀時直接return,這樣就會導致最終的動畫和GIF相比可能被截斷。反復試驗,找了個相對平衡的點,既保證單張圖片的清晰度,也保證整體的完整性,但這個方案不夠健壯,會隨著GIF圖的變化出現不同的問題。

下面介紹下上面說的這個方案,原理上基本清晰,就是通過ViewFlipper,向其中動態添加ImageView,每一個ImageView加載一幀圖片,從而達到動畫效果。

val viewFlipper = RemoteViews(context.packageName, R.layout.sign_in_view_flipper)
var allSize = 0
kotlin.run {
    frameBitmaps.forEachIndexed { index, it ->
        logger.d("allSize = $allSize, index = $index")
        if (index % 5 != 0) {
            return@forEachIndexed
        }
        val ivRemoteViews = RemoteViews(context.packageName, R.layout.sign_in_per_frame_bitmap_view)
        var bitmapSize = GifDownloadUtils.getBitmapSize(it)
        var bitmap = it
        val matrix = Matrix()
        var scale = 432f / bitmap.width
        logger.d("start, bitmapSize = $bitmapSize")
        matrix.setScale(scale, scale)
        while (bitmapSize >= GifDownloadUtils.MAX_WIDGET_BITMAP_MEMORY) {
            bitmap = Bitmap.createBitmap(bitmap, 0, 0, bitmap.width, bitmap.height, matrix, true)
            bitmapSize = GifDownloadUtils.getBitmapSize(bitmap)
            logger.d("bitmapSize = $bitmapSize, scale = $scale")
            scale /= 2f
            matrix.setScale(scale, scale)
        }
        allSize += bitmapSize
        logger.d("allSize = $allSize")
        if (allSize >= GifDownloadUtils.maxTotalWidgetBitmapMemory()) {
            return@run
        }
        ivRemoteViews.setImageViewBitmap(R.id.iv_per_frame, bitmap)
        viewFlipper.addView(R.id.view_flipper, ivRemoteViews)
    }
}
  
logger.d("addView")
// 這里是由于addView添加的View都會顯示在最上面,所以這里通過在原卡片中添加相同id的view,先把原卡的移除,再把新建的添加進去,達到更新的效果,這樣布局的層級就還是原先的層級。
remoteViews.removeAllViews(R.id.view_flipper)
remoteViews.addView(R.id.view_flipper, viewFlipper)

其中frameBitmaps就是上面獲得的所有圖片。

到這里網絡GIF圖片的加載也基本完成了。

04、總  結

上述提出的加載網絡GIF的方案,雖然解決了現有方案中加載GIF需要引入很多圖片資源或者GIF資源,導致包體大小增加的問題,但是如果GIF圖片本身質量較高,通過新方案可能會降低GIF的質量。

上述三種方案的優缺點和適用場景總結如下:



方案





優點





缺點





適用場景





網絡GIF加載





不引入本地資源,各Android版本兼容性好





1. 需要聯網權限,消耗流量

2. GIF過大時采樣會降低原圖質量

實現方式復雜





對包體大小增長極其敏感的場景





ViewFlipper實現





各Android版本兼容性好,實現方式簡單





1. ImageView過多,代碼過多,修改替換麻煩

2. 引入資源過多,包體增大

3. 在Remoteviews中,ViewFlipper的很多方法無法使用,比如停止播放等





GIF相對簡單或對包體大小增長不太敏感的場景





AnimatedImageDrawable實現





資源少,替換方便,實現方式簡單





1. 只支持Android9以上的系統才可以使用

2. GIF資源會帶來包體大小增長





GIF相對簡單或對包體大小增長不太敏感的場景且大于Android9的系統



總而言之,具體采用哪種方案需要根據實際開發的具體需要來實現,綜合方案的優缺點和適用場景來選擇。

責任編輯:龐桂玉 來源: vivo互聯網技術
相關推薦

2012-01-13 11:10:12

2015-06-25 11:20:01

開發者云服務APP

2012-01-09 14:17:44

Android應用商店

2009-10-16 10:08:14

2019-05-27 08:27:35

操作系統物聯網IOT

2013-08-06 13:30:15

管理員桌面管理員

2009-10-29 08:45:48

Windows 7版本升級

2020-10-15 09:33:48

網盤

2011-01-26 09:50:18

Silverlight.NetJavaScript

2014-07-02 18:27:56

2020-08-11 13:30:35

華為智慧交通懂行懂行

2017-03-28 13:00:10

LinuxGifineGIF動畫

2013-07-29 11:13:32

2009-07-12 10:11:27

2016-03-04 13:45:56

七牛云

2012-06-13 11:01:59

英特爾

2011-04-20 09:27:32

Spring

2011-08-29 17:17:00

Android應用gif快手iPhone應用

2013-09-25 10:06:49

桌面虛擬化Persistent

2022-08-03 09:58:03

跨端框架實踐
點贊
收藏

51CTO技術棧公眾號

日韩专区在线播放| 伊人精品影院| 国产麻豆一区| 欧美日韩国产一区精品一区| 午夜精品久久久久久久蜜桃app| 国产精品国内视频| 午夜不卡久久精品无码免费| 一区二区三区日| 欧美巨大xxxx| 亚洲天堂免费在线观看视频| 国产经典一区二区| 国产肉体xxxx裸体784大胆| 91一区二区三区在线| 懂色av中文字幕一区二区三区| 色yeye香蕉凹凸一区二区av| 99热成人精品热久久66| 欧洲精品久久一区二区| 欧美激情亚洲| 亚洲欧美精品在线| 午夜肉伦伦影院| 美女国产在线| 国产一区二区网址| 日韩有码视频在线| av黄色一级片| 国产精品论坛| 99热在这里有精品免费| 欧美精品国产精品日韩精品| 最新国产精品自拍| 日韩特级毛片| 国产成人免费视频精品含羞草妖精| 日韩一区二区欧美| 国产传媒第一页| 一区二区精彩视频| 一区二区三区欧洲区| 精品久久久久久亚洲精品| 99免费在线观看视频| 久久高清内射无套| 日韩午夜电影免费看| 国产精品系列在线| 国产日韩欧美日韩| 秋霞欧美一区二区三区视频免费| 国产精品亚洲d| 亚洲国产精品成人久久综合一区| 官网99热精品| 精品免费囯产一区二区三区| 欧美男gay| 欧美日韩一本到| 欧美日韩中文字幕在线播放| 国产综合视频在线| 亚洲日本激情| 国产亚洲精品综合一区91| 黄色手机在线视频| av中文字幕在线播放| 成人爽a毛片一区二区免费| 97成人超碰免| 粉嫩精品久久99综合一区| 欧美一区=区三区| 日本国产一区二区| 久久久久久久久久久久久国产| 亚洲第九十九页| 国产精品一区亚洲| 色999日韩欧美国产| 亚洲黄色小说视频| 日韩精品一区国产| 色噜噜久久综合| 欧美亚洲另类色图| 黄色精品免费看| 99久久国产综合精品女不卡| 国产精品美女www| 青青草原在线免费观看| 亚洲69av| 日韩一区二区免费电影| 日韩精品xxxx| gogogogo高清视频在线| 亚洲视频中文字幕| 91精品国产毛片武则天| 天天色天天射天天综合网| 亚洲蜜桃精久久久久久久| 狠狠色综合网站久久久久久久| 中文字幕第三页| 国产视频一区三区| 欧美综合第一页| 久久久国产成人| 成人三级视频| 精品亚洲一区二区三区在线观看| 在线一区二区不卡| 奇米777日韩| 亚洲夂夂婷婷色拍ww47| 亚洲精品在线视频观看| 四虎永久在线精品免费网址| 黄网站免费久久| 国产精品扒开腿做爽爽爽视频| 国产偷人爽久久久久久老妇app | 精品一区在线播放| 欧美高清成人| 岛国一区二区三区| 精品欧美一区二区久久久伦| 国产日韩精品在线看| 成年人网站91| 成人黄色生活片| 成人毛片一区二区三区| 国产精品毛片一区二区三区| 国产精品成人一区二区| 精品国产99久久久久久宅男i| 亚洲国产专区| 国产精品com| 亚洲AV无码精品色毛片浪潮| 91麻豆产精品久久久久久| 高清不卡日本v二区在线| 污视频网站免费观看| 国产69精品久久久久毛片| 麻豆蜜桃91| 丰满少妇一级片| 久久精品人人做人人综合| 久久久久无码国产精品一区| 神马久久久久久久久久| 国产精品情趣视频| 亚洲乱码国产乱码精品天美传媒| 手机av在线播放| 在线观看av一区| 日韩大片一区二区| 另类ts人妖一区二区三区| 日韩www在线| 一级性生活毛片| 久久综合亚洲| 久久久久国产精品www| 中文字幕男人天堂| 久久久亚洲午夜电影| 少妇免费毛片久久久久久久久| 国产高清av在线| 午夜欧美视频在线观看| 亚洲成人av免费观看| 91欧美日韩在线| 亚洲第一偷拍网| 亚洲av无码一区二区二三区| 欧美国产日本| 欧美一区二区三区四区在线| 无码一区二区三区在线观看| 奇米在线7777在线精品| 国产色婷婷国产综合在线理论片a| 五月激情婷婷综合| 亚洲一区在线看| 国产l精品国产亚洲区久久| 伊人精品久久| 色综合久久悠悠| 精品国产乱码久久久久久鸭王1| 日韩高清欧美激情| 91网站在线看| 天堂中文资源在线观看| 一区二区三区中文字幕精品精品 | 在线综合亚洲| 国产成人综合精品在线| 三级国产在线观看| 国产精品国产三级国产有无不卡 | 久久视频一区二区三区| 综合天天久久| 国产91|九色| 涩涩视频在线观看免费| 香蕉影视欧美成人| 艳妇乳肉豪妇荡乳xxx| 伊人成人在线视频| 国产精品久久久久999| 成人av电影观看| 亚洲综合色区另类av| 中国特级黄色片| 成人嘿咻视频免费看| 国产精品久久久久久五月尺| 在线免费黄色| 欧美日韩亚洲精品一区二区三区| 伊人影院综合在线| 欧美日韩看看2015永久免费| 8x拔播拔播x8国产精品| 欧美精品少妇| 欧美色倩网站大全免费| 手机av在线看| 成人小视频免费观看| 欧美综合亚洲图片综合区| 在线成人性视频| av电影高清在线观看| 欧美一区二区人人喊爽| 性の欲びの女javhd| 国产综合自拍| 国产在线精品播放| www红色一片_亚洲成a人片在线观看_| 日韩精品在线一区| 亚洲aaa视频| 久久婷婷丁香| 精品久久中出| 日韩制服一区| 欧美理论片在线观看| 天堂网av2014| 欧美日韩在线一区二区| 黄色一级免费视频| 久久99精品国产| 欧美一区二区视频17c| 成人福利影视| 日韩免费视频一区| 国产尤物在线视频| 国产91精品免费| www.com毛片| 偷偷www综合久久久久久久| 人体精品一二三区| 老司机av在线免费看| 精品国产免费久久| 欧美交换国产一区内射| 久久久久久毛片| 91免费视频污| 久久午夜影视| 成人免费性视频| 电影91久久久| www国产亚洲精品久久网站| 丰满人妻一区二区三区免费| 色综合久久综合中文综合网| av无码一区二区三区| 日韩视频久久| 在线观看免费91| 日韩有码中文字幕在线| 欧美一区二区三区免费视| 国产激情在线视频| 亚洲欧美中文字幕在线一区| 特级毛片www| 一区二区三区在线免费观看| 亚洲综合欧美综合| av亚洲产国偷v产偷v自拍| 色网站在线视频| 日韩av中文在线观看| 日韩欧美国产电影| 丰满少妇一区二区三区| 国产剧情一区在线| 男女h黄动漫啪啪无遮挡软件| 欧美日韩亚洲国产| 久久久久久国产精品美女| 免费观看久久久久| 亚洲片av在线| 日韩有码电影| 日韩经典一区二区三区| 乱精品一区字幕二区| 日韩一区二区视频在线观看| 综合久久中文字幕| 日本精品一区二区三区高清| 日韩av大片在线观看| 亚洲第一精品在线| 岛国精品资源网站| 国产宾馆实践打屁股91| 亚洲综合伊人久久| 国内精品伊人久久久久影院对白| 丁香婷婷激情网| 亚洲国产精品成人| 国产丝袜不卡| 成人一区视频| 久久久久久久久国产| 菠萝菠萝蜜在线视频免费观看| 视频在线观看99| av在线免费观看网| 中文字幕在线精品| 午夜精品久久久久久久第一页按摩 | 欧美乱偷一区二区三区在线| 欧美13videosex性极品| 亚洲欧美另类中文字幕| 天天操天天舔天天干| 日韩av综合网站| 撸视在线观看免费视频| 亚洲欧美日韩直播| www在线免费观看| 日韩在线视频免费观看| 免费黄网站在线播放| 九九视频这里只有精品| 91av久久| 久久精品视频在线| 天堂影院在线| 欧美一区二区精品在线| 亚洲av永久纯肉无码精品动漫| 欧美tk丨vk视频| 在线观看一二三区| 欧美日韩一区二区三区| 五月婷婷亚洲综合| 欧美午夜不卡在线观看免费| 日本少妇毛茸茸高潮| 亚洲欧洲日产国产综合网| 波多野结衣福利| 久久精品人人做人人综合| 最新黄色av网址| 亚洲一区二区三区小说| 超碰超碰超碰超碰| 欧美日韩国产片| 秋霞av一区二区三区| 欧美视频一区二区三区四区| 免费高清视频在线观看| 国产精品国精产品一二| 欧美精品videos| 综合在线影院| 亚洲va欧美va国产综合剧情| 久久99久久99精品免观看软件| 久久久久久久色| 欧美电影免费观看高清完整| 国产自摸综合网| 日韩精品社区| 男女啪啪的视频| 欧美黄色大片在线观看| 日韩高清在线播放| 免费av一区| dy888午夜| 噜噜噜在线观看免费视频日韩| 少妇高潮毛片色欲ava片| 欧美日韩国产精品一区二区亚洲| 久久精品国产精品亚洲色婷婷| 精品一区二区在线免费观看| 久久福利小视频| 18欧美乱大交hd1984| 亚洲欧美综合自拍| 大伊人狠狠躁夜夜躁av一区| 91超薄丝袜肉丝一区二区| 日韩av网站大全| 97影院秋霞午夜在线观看| 国产激情综合五月久久| 国产精品乱战久久久| 国产精品二区三区| 欧美色就是色| 日韩在线综合网| 国产激情偷乱视频一区二区三区| 国产精成人品免费观看| 亚洲va天堂va国产va久| 国产又粗又猛又爽又黄视频| 91精品国产综合久久福利软件| 奇米影视888狠狠狠777不卡| 色综合久久悠悠| 国产精品一区二区三区www| 日产精品一线二线三线芒果| 亚洲国产裸拍裸体视频在线观看乱了中文 | 1区2区3区在线观看| 91国语精品自产拍在线观看性色| 高清不卡一区| 亚洲天堂电影网| 久久精品国产www456c0m| 欧美午夜性视频| 国产精品一区二区久久精品爱涩| 天堂网av2018| 欧美三级欧美一级| 国产在线视频福利| 精品久久久999| 国产精品成人国产| 日韩av不卡播放| 日精品一区二区三区| 添女人荫蒂视频| 欧美日韩亚洲激情| 亚洲aaa在线观看| 69**夜色精品国产69乱| 全球av集中精品导航福利| 亚洲理论电影在线观看| 国产精品99久久久久久似苏梦涵| 韩国一级黄色录像| 69久久99精品久久久久婷婷 | 中国字幕a在线看韩国电影| 国产成人精品久久二区二区91| 欧美激情网址| 无码aⅴ精品一区二区三区浪潮 | 中文字幕资源网在线观看免费 | 亚洲性感美女99在线| 91成人在线观看喷潮蘑菇| 亚洲综合色在线| 日本波多野结衣在线| 欧美亚洲在线观看| 国产精品亚洲二区| 欧美日韩dvd| 国产成人亚洲综合a∨婷婷 | 亚洲精品人人| 丰满少妇一区二区| 在线亚洲+欧美+日本专区| 91涩漫在线观看| 91日韩久久| av亚洲在线观看| 欧美日韩一道本| 久久精品欧美日韩| 国产精品乱码久久久| 欧美福利视频网站| 欧美wwwwww| 在线看的黄色网址| 一区二区免费看| 午夜视频福利在线| 国产精品久久久久av| 亚洲自拍偷拍网| 91av在线免费| 91精品国产综合久久精品app| av资源在线| 亚洲黄色成人久久久| 国产精品一区一区| 亚洲天堂视频网站| 久久久久www| 亚洲伊人春色| 五月六月丁香婷婷| 欧美特级www| 欧美一级性视频| 国产精品久久不能| 欧美日韩亚洲三区| 亚洲色成人网站www永久四虎| 日韩午夜激情视频| 欧美色网在线| 国产黄色片免费在线观看| 中文幕一区二区三区久久蜜桃| 不卡av中文字幕|