簡(jiǎn)單實(shí)現(xiàn)android輪播圖
輪播圖是很常用的一個(gè)效果 核心功能已經(jīng)實(shí)現(xiàn) 沒(méi)有什么特殊需求 自己沒(méi)事研究的 所以封裝的不太好 一些地方還比較糙 為想要研究輪播圖的同學(xué)提供個(gè)參考
目前測(cè)試圖片為mipmap中的圖片 沒(méi)有寫(xiě)從網(wǎng)絡(luò)加載圖片 可自行根據(jù)需求在getShowView()方法中修改
1.定時(shí)切換
通過(guò)handle延時(shí)發(fā)送通知改變界面 然后在切換viewpage的界面之后 再次發(fā)送此延時(shí)通知 就ok咯 還可以通過(guò)timer定時(shí)器實(shí)現(xiàn)
2.無(wú)限輪播效果
如果我們只是在自動(dòng)輪播到最后一頁(yè) 然后進(jìn)行判斷讓切換到第一頁(yè) 這樣是可以實(shí)現(xiàn)輪播的效果
但是 有兩個(gè)問(wèn)題
- 切換從最后一頁(yè)切換到第一頁(yè)的時(shí)候有一個(gè)很明顯的回滾效果 不是我們想要的
- 當(dāng)我們手動(dòng)滑動(dòng)的時(shí)候 在第一頁(yè)和最后一頁(yè)的時(shí)候 無(wú)法繼續(xù)左右滑動(dòng) 因?yàn)橐呀?jīng)沒(méi)有下一頁(yè)了
先看張圖(偷來(lái)的)
不得不說(shuō)這位兄弟的圖p的很形象 簡(jiǎn)直完美
雖然看到的是三張圖 實(shí)際上是五張 數(shù)據(jù)多的時(shí)候也按照這種方式添加數(shù)據(jù) 當(dāng)view4的時(shí)候自動(dòng)切換到view5時(shí) 進(jìn)行判斷讓到切換到view2 這樣造成的感覺(jué)就是最后一張下來(lái)是第一張
我們利用viewpage自帶的方法切換界面立即切換沒(méi)有滾動(dòng)效果 當(dāng)圖片一樣的時(shí)候是看不出圖片變化的
setCurrentItem(int item, boolean smoothScroll)
第二個(gè)參數(shù)設(shè)置false 界面切換的時(shí)候無(wú)滾動(dòng)效果 默認(rèn)true
好啦 接下來(lái)看代碼
public class BannerViewPager extends FrameLayout { private ViewPager viewPager; private TextView tvTitle; private LinearLayout indicatorGroup; private BannerAdapter adapter; private List<String> titles;//標(biāo)題集合 private List imageUrls;//圖片數(shù)據(jù) private List<View> views;//輪播圖顯示 private ImageView [] tips;//保存顯示的小圓點(diǎn) private int count;//保存imageUrls的總數(shù) private int bannerTime=2500;//輪播圖的間隔時(shí)間 private int currentItem=0;//輪播圖的當(dāng)前選中項(xiàng) private long releaseTime = 0;//保存觸發(fā)時(shí)手動(dòng)滑動(dòng)的時(shí)間 進(jìn)行判斷防止滑動(dòng)之后立即輪播 private final int START=10; private final int STOP=20; private Context context; private Handler handler; private final Runnable runnable=new Runnable() { @Override public void run() { long now=System.currentTimeMillis(); if (now-releaseTime>bannerTime-500){ handler.sendEmptyMessage(START); }else{ handler.sendEmptyMessage(STOP); } } }; public BannerViewPager(Context context) { super(context); } public BannerViewPager(Context context, AttributeSet attrs) { super(context, attrs); this.context=context; titles=new ArrayList<>(); titles.add("標(biāo)題1"); titles.add("標(biāo)題2"); titles.add("標(biāo)題3"); imageUrls=new ArrayList(); views=new ArrayList<>(); init(context,attrs); } private void init(final Context context, AttributeSet attrs){ View view= LayoutInflater.from(context).inflate(R.layout.layout_banner,this); viewPager= (ViewPager) view.findViewById(R.id.banner_view_pager); tvTitle= (TextView) view.findViewById(R.id.banner_title); indicatorGroup= (LinearLayout) view.findViewById(R.id.banner_indicator); handler=new Handler(){ @Override public void handleMessage(Message msg) { super.handleMessage(msg); switch (msg.what){ case START: viewPager.setCurrentItem(currentItem+1); handler.removeCallbacks(runnable); handler.postDelayed(runnable,bannerTime); break; case STOP: releaseTime=0; handler.removeCallbacks(runnable); handler.postDelayed(runnable,bannerTime); break; } } }; } /** * 初始化數(shù)據(jù) 以及拿到數(shù)據(jù)后的各種設(shè)置 * 可以是網(wǎng)絡(luò)地址 也可是項(xiàng)目圖片數(shù)據(jù) * @param imageUrls */ public void setData(List<?> imageUrls){ this.imageUrls.clear(); this.count=imageUrls.size(); this.imageUrls.add(imageUrls.get(count-1)); this.imageUrls.addAll(imageUrls); this.imageUrls.add(imageUrls.get(0)); initIndicator(); getShowView(); setUI(); } /** * 設(shè)置標(biāo)題 * @param titles */ public void setTitles(List<String> titles){ this.titles.clear(); this.titles.addAll(titles); } /** * 設(shè)置小圓點(diǎn)指示器 */ private void initIndicator(){ tips=new ImageView[count]; LinearLayout.LayoutParams layoutParams = new LinearLayout. LayoutParams(new ViewGroup.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT)); layoutParams.height=10; layoutParams.width=10; layoutParams.leftMargin = 5;// 設(shè)置點(diǎn)點(diǎn)點(diǎn)view的左邊距 layoutParams.rightMargin = 5;// 設(shè)置點(diǎn)點(diǎn)點(diǎn)view的右邊距 for (int i=0;i<count;i++){ ImageView imageView=new ImageView(context); if (i == 0) { imageView.setBackgroundResource(R.drawable.shape_circle_red); } else { imageView.setBackgroundResource(R.drawable.shape_circle_white); } tips[i] = imageView; indicatorGroup.addView(imageView, layoutParams); } } /** * 獲取顯示圖片view */ private void getShowView(){ for (int i=0;i<imageUrls.size();i++){ ImageView imageView=new ImageView(context); imageView.setScaleType(ImageView.ScaleType.CENTER_CROP); if (imageUrls.get(i) instanceof String){ }else{ imageView.setImageResource((Integer) imageUrls.get(i)); } views.add(imageView); } } /** * 設(shè)置UI */ private void setUI(){ adapter=new BannerAdapter(); viewPager.setAdapter(adapter); viewPager.addOnPageChangeListener(onPageChangeLis); viewPager.setCurrentItem(1); handler.postDelayed(runnable,bannerTime); } /** * viewPage改變監(jiān)聽(tīng) */ private ViewPager.OnPageChangeListener onPageChangeLis=new ViewPager.OnPageChangeListener() { @Override public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) { } @Override public void onPageSelected(int position) { //計(jì)算當(dāng)前頁(yè)的下標(biāo) int max = views.size() - 1; int temp = position; currentItem = position; if (position == 0) { currentItem = max - 1; } else if (position == max) { currentItem = 1; } temp = currentItem - 1; setIndicatorAndTitle(temp); } @Override public void onPageScrollStateChanged(int state) { currentItem=viewPager.getCurrentItem(); switch (state) { case 0: //Log.e("aaa","=====靜止?fàn)顟B(tài)======"); if (currentItem == 0) { viewPager.setCurrentItem(count, false); } else if (currentItem == count + 1) { viewPager.setCurrentItem(1, false); } break; case 1: // Log.e("aaa","=======手動(dòng)拖拽滑動(dòng)時(shí)調(diào)用===="); releaseTime = System.currentTimeMillis(); if (currentItem == count + 1) { viewPager.setCurrentItem(1, false); } else if (currentItem == 0) { viewPager.setCurrentItem(count, false); } break; case 2: // Log.e("aaa","=======自動(dòng)滑動(dòng)時(shí)調(diào)用===="); break; } } }; /** * 設(shè)置指示器和標(biāo)題切換 * @param position */ private void setIndicatorAndTitle(int position){ tvTitle.setText(titles.get(position)); for (int i=0;i<tips.length;i++){ if (i==position){ tips[i].setBackgroundResource(R.drawable.shape_circle_red); }else{ tips[i].setBackgroundResource(R.drawable.shape_circle_white); } } } /** * 適配器 */ class BannerAdapter extends PagerAdapter{ @Override public int getCount() { return views.size(); } @Override public boolean isViewFromObject(View view, Object object) { return view==object; } @Override public Object instantiateItem(ViewGroup container, int position) { container.addView(views.get(position)); return views.get(position); } @Override public void destroyItem(ViewGroup container, int position, Object object) { container.removeView((View) object); } } }
Activity代碼
BannerViewPager banner= (BannerViewPager) findViewById(R.id.banner); List<Integer> imageUrl=new ArrayList<>(); imageUrl.add(R.mipmap.aiyo); imageUrl.add(R.mipmap.dipang1); imageUrl.add(R.mipmap.ic_launcher); banner.setData(imageUrl);
最后提供兩個(gè)github上大神封裝好的輪播圖
建議不太會(huì)的同學(xué)先搞清楚基本的邏輯在使用第三方庫(kù)
https://github.com/youth5201314/banner
https://github.com/bingoogolapple/BGABanner-Android
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Android實(shí)現(xiàn)輪播圖片展示效果
- Android實(shí)現(xiàn)炫酷輪播圖效果
- Android使用viewpager實(shí)現(xiàn)自動(dòng)無(wú)限輪播圖
- Android實(shí)現(xiàn)ViewPage輪播圖效果
- Android ViewPager實(shí)現(xiàn)輪播圖效果
- Android開(kāi)發(fā)實(shí)現(xiàn)的自動(dòng)換圖片、輪播圖效果示例
- Android如何使用RecyclerView打造首頁(yè)輪播圖
- android實(shí)現(xiàn)banner輪播圖無(wú)限輪播效果
- Android自定義控件實(shí)現(xiàn)優(yōu)雅的廣告輪播圖
- Android自定義輪播圖效果
相關(guān)文章
Android程序開(kāi)發(fā)之使用Design包實(shí)現(xiàn)QQ動(dòng)畫(huà)側(cè)滑效果和滑動(dòng)菜單導(dǎo)航
這篇文章主要介紹了Android程序開(kāi)發(fā)之使用Design包實(shí)現(xiàn)QQ動(dòng)畫(huà)側(cè)滑效果和滑動(dòng)菜單導(dǎo)航的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-07-07Android 藍(lán)牙自動(dòng)匹配PIN碼跳過(guò)用戶(hù)交互示例
本篇文章主要介紹了Android 藍(lán)牙自動(dòng)匹配PIN碼跳過(guò)用戶(hù)交互示例,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-06-06Android編程獲取Wifi名稱(chēng)(SSID)的方法
這篇文章主要介紹了Android編程獲取Wifi名稱(chēng)(SSID)的方法,涉及Android基于WifiManager和WifiInfo操作Wifi信息的相關(guān)實(shí)現(xiàn)技巧,需要的朋友可以參考下2017-05-05Android Studio綁定下拉框數(shù)據(jù)詳解
這篇文章主要為大家詳細(xì)介紹了Android Studio綁定下拉框數(shù)據(jù),Android Studio綁定網(wǎng)絡(luò)JSON數(shù)據(jù),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-10-10基于Android實(shí)現(xiàn)跳轉(zhuǎn)到WiFi開(kāi)關(guān)設(shè)置頁(yè)的詳細(xì)步驟
在Android應(yīng)用開(kāi)發(fā)中,有時(shí)候需要引導(dǎo)用戶(hù)到特定的系統(tǒng)設(shè)置頁(yè)面,例如Wi-Fi開(kāi)關(guān)設(shè)置頁(yè),可以通過(guò)隱式Intent來(lái)實(shí)現(xiàn)這一功能,以下是詳細(xì)的步驟以及相關(guān)的Kotlin代碼示例,需要的朋友可以參考下2024-09-09android webvie指定視頻播放器播放網(wǎng)站視頻
android webview過(guò)濾調(diào)用第三方瀏覽器,并且解析視頻網(wǎng)站播放地址,使用指定播放器2013-11-11