Android3D畫廊效果與自動輪播Banner

最開始項目需要使用3D畫廊效果作為首頁輪播,網上找了半天也沒有比較滿意的,最終決定自己寫一個。本控件采用的是viewpager完成的,支持無限滑動的3D視覺的畫廊效果、 平面普通廣告欄輪播。提供對外方法:指示器圖片自定義、小圖片位置、是否圓角等。本代碼已托管到[github]https://github.com/lzjin/ViewPagerGallery

1、先看效果圖:

gif1.gif

gif2.gif

ic_banner3.png

ic_banner1.png

ic_banner2.png

2、效果分析 3D畫廊效果

代碼調用:

mViewPager.initBanner(imagesUriList, true)//圖片地址,isGallery參數是否開啟3D畫廊效果
   .addPageMargin(10, 50)//參數1page之間的間距,參數2中間item距離邊界的間距
   .addPoint(6)//添加指示器
   .addPointBottom(7)
   .addStartTimer(5)//自動輪播5秒間隔
   .addRoundCorners(12)//圓角
   .finishConfig()//這句必須加
   .addBannerListener(new BannerViewPager.OnClickBannerListener() {
    @Override
    public void onBannerClick(int position) {
          //點擊item
    }
});

代碼分析:

3D畫廊效果,是通過滑動的屬性動畫來設置。那我們就得就得了解ViewPager的PageTransformer類。重寫PageTransformer,在滑動的時候進行X軸、Y軸的縮放拉伸來實現。
(1)(-oo,-1) 相對于左邊第一頁,其左邊的所有頁面

if (position < -1) {
   view.setScaleX(MIN_SCALE);
   view.setScaleY(MIN_SCALE);
   view.setAlpha(MIN_ALPHA);//這里是設置透明度
}

(2)[-1, 1 )當前頁的左右第一頁

else if (position < 1) {
     float scaleFactor = MIN_SCALE + (1 - Math.abs(position)) * (MAX_SCALE - MIN_SCALE);
     //[0, 1 ) 相對于當前選中頁,其右邊第一頁 **
     if (position > 0) {
          view.setTranslationX(-scaleFactor);
     }
     // [-1, 0 ) 相對于當前選中頁,其左邊的第一頁**
     else if (position < 0) {
          view.setTranslationX(scaleFactor);
     }
     view.setScaleY(scaleFactor);
     view.setScaleX(scaleFactor);
     // float alpha = 1f -  Math.abs(position) * (1 - );
     float alpha = MIN_ALPHA + (1 - MIN_ALPHA) * (1 - Math.abs(position));
     view.setAlpha(alpha);//透明度  
}

(3)[1,+oo) 相對于右邊第一頁,其右邊的所有頁面

// (1,+Infinity]
else { 
    view.setScaleX(MIN_SCALE);
    view.setScaleY(MIN_SCALE);
    view.setAlpha(MIN_ALPHA);//透明度
}

(4)為了讓界面顯示3item數據,設置左右間距,這里要注意以下 android:clipChildren="false" 這句代碼的含義,就是不限制View的布局,已達到邊界繪制效果。(間距根據自己需要可更改)

<android.support.v4.view.ViewPager
  android:id="@+id/viewPager"
  android:layout_width="match_parent"
  android:layout_height="wrap_content"
  android:clipChildren="false"
  android:layout_marginLeft="60dp"
  android:layout_marginRight="60dp"/>

(5)圓角設置
由于網絡圖片加載我用的Glide庫,對應的圓角設置重寫BitmapTransformation即可。這里不多講,網上工具類很多。

//自定義圓角
public class CornerTransform extends BitmapTransformation {
 
    private static float radius = 0f;
 
    public CornerTransform(Context context) {
        this(context, 4);
    }
 
    public CornerTransform(Context context, int dp) {
        super(context);
        this.radius = Resources.getSystem().getDisplayMetrics().density * dp;
    }
 
    @Override protected Bitmap transform(BitmapPool pool, Bitmap toTransform, int outWidth, int outHeight) {
        return roundCrop(pool, toTransform);
    }
 
    private static Bitmap roundCrop(BitmapPool pool, Bitmap source) {
        if (source == null) return null;
 
        Bitmap result = pool.get(source.getWidth(), source.getHeight(), Bitmap.Config.ARGB_8888);
        if (result == null) {
            result = Bitmap.createBitmap(source.getWidth(), source.getHeight(), Bitmap.Config.ARGB_8888);
        }
 
        Canvas canvas = new Canvas(result);
        Paint paint = new Paint();
        paint.setShader(new BitmapShader(source, BitmapShader.TileMode.CLAMP, BitmapShader.TileMode.CLAMP));
        paint.setAntiAlias(true);
        RectF rectF = new RectF(0f, 0f, source.getWidth(), source.getHeight());
        canvas.drawRoundRect(rectF, radius, radius, paint);
        return result;
    }
    @Override public String getId() {
        return getClass().getName() + Math.round(radius);
    }
}

3、 普通效果2

跟上面的實現差不多,唯一的區別就是不重寫PageTransformer,參數設置false,使用系統默認的滑動效果即可。

mViewPager.initBanner(imagesUriList, false)//圖片地址,關閉3D畫廊效果
   .addPageMargin(10, 50)//參數1page之間的間距,參數2中間item距離邊界的間距
   .addPoint(5)//添加指示器,5dp
   .addPointBottom(7)
   .addStartTimer(5)//自動輪播5秒間隔
   .addRoundCorners(12)//圓角
   .finishConfig()//這句必須加
   .addBannerListener(new BannerViewPager.OnClickBannerListener() {
    @Override
    public void onBannerClick(int position) {
          //點擊item
    }
});

4、 普通效果1

在普通效果2上,少了圓角設置,已經左右間距設置0即可。

mViewPager.initBanner(imagesUriList, false)//圖片地址,關閉3D畫廊效果
   .addPageMargin(0, 0)//無間距
   .addPoint(5)//添加指示器,5dp
   .addPointBottom(7)
   .addStartTimer(5)//自動輪播5秒間隔
   .finishConfig()//這句必須加
   .addBannerListener(new BannerViewPager.OnClickBannerListener() {
    @Override
    public void onBannerClick(int position) {
          //點擊item
    }
});

5、 方法講解

initBanner(List<String> imagesUriList,boolean isGallery)//圖片,是否開啟3D畫廊效果
initBanner(List<String> imagesUriList,boolean isGallery,float alpha)//圖片,畫廊,alpha透明度
addPoint(int distance) //間距
addPoint(int distance,int piont_press,int piont) //間距、選中小圓點自定義、未選中小圓點自定義
addRoundCorners(int corners)//圓角10dp
addStartTimer()、stopTimer()
addDefaultImg()
//注意:當添加了3D畫廊效果時,columnMargin盡量設小。應該本是已經進行了x、y的縮放
addPageMargin(int columnMargin,int rowMargin)//兩個Page之間的距離,中間item的對邊界的邊距

6.結尾

因為是網絡圖片,請加網絡權限。
如果還是有什么不懂,請前往github查看源碼。
純手工寫,實屬不易,各位看官如果本文對你有幫助,請點個贊吧。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,363評論 6 532
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,497評論 3 416
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,305評論 0 374
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,962評論 1 311
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,727評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,193評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,257評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,411評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,945評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,777評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,978評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,519評論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,216評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,642評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,878評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,657評論 3 391
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,960評論 2 373

推薦閱讀更多精彩內容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,643評論 25 708
  • 1、通過CocoaPods安裝項目名稱項目信息 AFNetworking網絡請求組件 FMDB本地數據庫組件 SD...
    陽明先生_X自主閱讀 16,000評論 3 119
  • 用兩張圖告訴你,為什么你的 App 會卡頓? - Android - 掘金 Cover 有什么料? 從這篇文章中你...
    hw1212閱讀 12,777評論 2 59
  • https://developer.apple.com/library/mac/documentation/Swi...
    ChiOS閱讀 156評論 0 0
  • 眼里的小傻白甜也開始,人呀,起來了!肯定是長大了些許!不能老是傻白甜,那樣不好![偷笑]
    縱情嬉戲天地間閱讀 82評論 0 0