Android开发,使用ViewPager2实现自动轮播图

文章目录

    • 1. build.gradle添加依赖:
    • 2. AndroidManifest.xml中添加网络访问权限
    • 3. 编写 布局文件
    • 4. 编写Banner适配器
    • 5. 自定义Banner视图
    • 6. 定义圆点指示器的drawable资源
    • 7. 在需要使用轮播图中的Activity中使用
    • 8. 运行效果图
    • 9. 视频教程

在Android项目程序设计中,首页轮播图是个很常见的功能,在之前的文章中,教了大家如何使用第三方库来快速实现轮播图(教程地址在文章末尾给出),今天教大家手把手来撸一个轮播图

1. build.gradle添加依赖:

图片显示,采用Glide来加载网络图片

implementation 'com.github.bumptech.glide:glide:4.12.0'

2. AndroidManifest.xml中添加网络访问权限

 <uses-permission android:name="android.permission.INTERNET" />

3. 编写 布局文件

activity_banner.xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="match_parent"android:layout_height="200dp"><androidx.viewpager2.widget.ViewPager2android:id="@+id/viewPager"android:layout_width="match_parent"android:layout_height="match_parent" /><LinearLayoutandroid:id="@+id/indicator"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_alignParentBottom="true"android:layout_centerHorizontal="true"android:layout_marginBottom="10dp"android:orientation="horizontal" /></RelativeLayout>

4. 编写Banner适配器

BannerAdapter.java

public class BannerAdapter extends RecyclerView.Adapter<BannerAdapter.BannerViewHolder> {private List<String> imageUrls;private Context context;public BannerAdapter(Context context, List<String> imageUrls) {this.context = context;this.imageUrls = imageUrls;}@NonNull@Overridepublic BannerViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {ImageView imageView = new ImageView(context);imageView.setLayoutParams(new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT,ViewGroup.LayoutParams.MATCH_PARENT));imageView.setScaleType(ImageView.ScaleType.CENTER_CROP);return new BannerViewHolder(imageView);}@Overridepublic void onBindViewHolder(@NonNull BannerViewHolder holder, int position) {// 使用实际位置加载图片int realPosition = position % imageUrls.size();// 这里使用Glide加载图片,需要添加Glide依赖Glide.with(context).load(imageUrls.get(realPosition)).into(holder.imageView);holder.imageView.setOnClickListener(v -> {// 处理图片点击事件if (onItemClickListener != null) {onItemClickListener.onItemClick(realPosition);}});}@Overridepublic int getItemCount() {// 返回一个很大的数,实现无限循环return Integer.MAX_VALUE;}static class BannerViewHolder extends RecyclerView.ViewHolder {ImageView imageView;public BannerViewHolder(@NonNull View itemView) {super(itemView);imageView = (ImageView) itemView;}}// 点击事件接口public interface OnItemClickListener {void onItemClick(int position);}private OnItemClickListener onItemClickListener;public void setOnItemClickListener(OnItemClickListener listener) {this.onItemClickListener = listener;}
}

这里注意:在方法onBindViewHolder中,Glide加载图片,需要添加Glide依赖

5. 自定义Banner视图

BannerView.java

public class BannerView extends RelativeLayout {private ViewPager2 viewPager;private LinearLayout indicator;private List<ImageView> indicatorDots;private Handler handler;private static final int SCROLL_TIME = 3000; // 自动滚动间隔private boolean isAutoScroll = true;public BannerView(Context context) {super(context);init(context);}public BannerView(Context context, AttributeSet attrs) {super(context, attrs);init(context);}private void init(Context context) {// 加载布局LayoutInflater.from(context).inflate(R.layout.activity_banner, this, true);viewPager = findViewById(R.id.viewPager);indicator = findViewById(R.id.indicator);indicatorDots = new ArrayList<>();handler = new Handler(Looper.getMainLooper());}public void setImages(List<String> imageUrls) {// 设置适配器BannerAdapter adapter = new BannerAdapter(getContext(), imageUrls);viewPager.setAdapter(adapter);// 设置初始位置到中间viewPager.setCurrentItem(Integer.MAX_VALUE / 2, false);// 创建指示器createIndicators(imageUrls.size());// 设置页面切换监听viewPager.registerOnPageChangeCallback(new ViewPager2.OnPageChangeCallback() {@Overridepublic void onPageSelected(int position) {updateIndicator(position % imageUrls.size());}});// 开始自动滚动startAutoScroll();}private void createIndicators(int count) {indicator.removeAllViews();indicatorDots.clear();for (int i = 0; i < count; i++) {ImageView dot = new ImageView(getContext());LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(dpToPx(8), dpToPx(8));params.setMargins(dpToPx(4), 0, dpToPx(4), 0);dot.setLayoutParams(params);dot.setImageResource(R.drawable.dot_normal);indicator.addView(dot);indicatorDots.add(dot);}// 设置第一个点为选中状态if (!indicatorDots.isEmpty()) {indicatorDots.get(0).setImageResource(R.drawable.dot_selected);}}private void updateIndicator(int position) {for (int i = 0; i < indicatorDots.size(); i++) {indicatorDots.get(i).setImageResource(i == position ? R.drawable.dot_selected : R.drawable.dot_normal);}}private void startAutoScroll() {handler.postDelayed(new Runnable() {@Overridepublic void run() {if (isAutoScroll && viewPager != null) {viewPager.setCurrentItem(viewPager.getCurrentItem() + 1);handler.postDelayed(this, SCROLL_TIME);}}}, SCROLL_TIME);}public void stopAutoScroll() {isAutoScroll = false;handler.removeCallbacksAndMessages(null);}private int dpToPx(int dp) {return (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, dp, getResources().getDisplayMetrics());}@Overrideprotected void onDetachedFromWindow() {super.onDetachedFromWindow();stopAutoScroll();}
}

6. 定义圆点指示器的drawable资源

  1. dot_normal.xml
<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android"android:shape="oval"><solid android:color="#80FFFFFF" /><sizeandroid:width="8dp"android:height="8dp" />
</shape>
  1. dot_selected.xml
<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android"android:shape="oval"><solid android:color="#FFFFFF" /><sizeandroid:width="8dp"android:height="8dp" />
</shape>

这里注意:dot_normal.xml和dot_selected.xml创建在res下的drawable中,而不是res下的layout

7. 在需要使用轮播图中的Activity中使用

  1. MainActivity.java
public class MainActivity extends AppCompatActivity {private BannerView bannerView;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);bannerView = findViewById(R.id.banner_view);// 准备图片URL列表List<String> imageUrls = new ArrayList<>();imageUrls.add("https://img0.baidu.com/it/u=2673431480,3550145287&fm=253&fmt=auto&app=138&f=JPEG?w=500&h=211");imageUrls.add("https://img1.baidu.com/it/u=1588549273,1647024151&fm=253&fmt=auto&app=138&f=JPEG?w=500&h=312");imageUrls.add("https://img2.baidu.com/it/u=3920432635,2105461872&fm=253&fmt=auto&app=138&f=JPEG?w=1600&h=500");imageUrls.add("https://img1.baidu.com/it/u=921161528,741350508&fm=253&fmt=auto&app=138&f=PNG?w=658&h=380");imageUrls.add("https://img2.baidu.com/it/u=611817443,624986336&fm=253&fmt=auto&app=138&f=JPEG?w=1280&h=407");// 设置图片bannerView.setImages(imageUrls);}@Overrideprotected void onDestroy() {super.onDestroy();if (bannerView != null) {bannerView.stopAutoScroll();}}
}

这里注意:一定要在Activity生命周期的onDestroy()中调用 bannerView.stopAutoScroll()方法

  1. activity_main.xml
<androidx.appcompat.widget.LinearLayoutCompat xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="match_parent"android:layout_height="200dp"><com.app.skill.sharing.views.BannerViewandroid:id="@+id/banner_view"android:layout_width="match_parent"android:layout_height="wrap_content" /></androidx.appcompat.widget.LinearLayoutCompat>

com.app.skill.sharing.views.BannerView :这里自定义控件的使用为:包名+类名。具体根据自己自定义控件存放的位置而定,在编辑器中输入Ban…,跟使用系统自带控件一样,编辑器会智能提示

8. 运行效果图

在这里插入图片描述

9. 视频教程

  1. Androidstudio轮播图Banner实现 :https://www.bilibili.com/video/BV18B4y1R7gy/?vd_source=984bb03f768809c7d33f20179343d8c8

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/888701.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

企业网双核心交换机实现冗余和负载均衡(MSTP+VRRP)

MSTP&#xff08;多生成树协议&#xff09; 通过创建多个VLAN实例&#xff0c;将原有的STP、RSTP升级&#xff0c;避免单一VLAN阻塞后导致带宽的浪费&#xff0c;通过将VLAN数据与实例绑定&#xff0c;有效提升网络速率。 VRRP&#xff08;虚拟路由冗余协议&#xff09; 用…

Socket编程-tcp

1. 前言 在tcp套接字编程这里&#xff0c;我们将完成两份代码&#xff0c;一份是基于tcp实现普通的对话&#xff0c;另一份加上业务&#xff0c;client输入要执行的命令&#xff0c;server将执行结果返回给client 2. tcp_echo_server 与udp类似&#xff0c;前两步&#xff1…

15分钟训练数字人MimicTalk

只需15分钟&#xff0c;就能训练高质量&#xff0c;个性化数字人大模型。由浙江大学与字节跳动联合推出MimicTalk算法&#xff0c;目前已开源。 在外表和说话风格上和真人相似。将通用3D数字人大模型适应到单个目标人&#xff0c;采用动静结合的高效微调方案&#xff0…

【kettle】mysql数据抽取至kafka/消费kafka数据存入mysql

目录 一、mysql数据抽取至kafka1、表输入2、json output3、kafka producer4、启动转换&#xff0c;查看是否可以消费 二、消费kafka数据存入mysql1、Kafka consumer2、Get records from stream3、字段选择4、JSON input5、表输出 一、mysql数据抽取至kafka 1、表输入 点击新建…

在 MacOS 上为 LM Studio 更换镜像源

在 MacOS 之中使用 LM Studio 部署本地 LLM时&#xff0c;用户可能会遇到无法下载模型的问题。 一般的解决方法是在 huggingface.co 或者国内的镜像站 hf-mirror.com 的项目介绍卡页面下载模型后拖入 LM Studio 的模型文件夹。这样无法利用 LM Studio 本身的搜索功能。 本文将…

vue中.sync修饰符的用法

一、什么是.sync修饰符 在Vue.js中&#xff0c;.sync 修饰符用于创建一个双向绑定的 prop。它使子组件能够更新父组件的 prop 值&#xff0c;实现父子组件之间的双向数据同步。具体来说&#xff0c;.sync 修饰符主要有以下几个功能&#xff1a; 简化双向绑定&#xff1a; 使用…

【附源码】基于环信鸿蒙IM SDK实现一个聊天Demo

项目背景 本项目基于环信IM 鸿蒙SDK 打造的鸿蒙IM Demo&#xff0c;完全适配HarmonyOS NEXT系统&#xff0c;实现了发送消息&#xff0c;添加好友等基础功能。代码开源&#xff0c;功能简洁&#xff0c;如果您有类似开发需求可以参考。 源码地址&#xff1a;https://github.c…

SHELL----正则表达式

一、文本搜索工具——grep grep -参数 条件 文件名 其中参数有以下&#xff1a; -i 忽略大小写 -c 统计匹配的行数 -v 取反&#xff0c;不显示匹配的行 -w 匹配单词 -E 等价于 egrep &#xff0c;即启用扩展正则表达式 -n 显示行号 -rl 将指定目录内的文件打…

Can‘t find variable: token(token is not defined)

文章目录 例子 1&#xff1a;使用 var例子 2&#xff1a;使用 let 或 const例子 3&#xff1a;异步操作你的代码中的情况 Cant find variable: tokentoken is not defined源代码 // index.jsPage({data: {products:[],cardLayout: grid, // 默认卡片布局为网格模式isGrid: tr…

【AI系统】GhostNet 系列

GhostNet 系列 本文主要会介绍 GhostNet 系列网络&#xff0c;在本文中会给大家带来卷积结构的改进方面的轻量化&#xff0c;以及与注意力(self-attention)模块的进行结合&#xff0c;部署更高效&#xff0c;更适合移动计算的 GhostNetV2。让读者更清楚的区别 V2 与 V1 之间的…

YOLOv8改进,YOLOv8引入CARAFE轻量级通用上采样算子,助力模型涨点

摘要 CARAFE模块的设计目的是在不增加计算复杂度的情况下,提升特征图的质量,特别是在视频超分辨率任务中,提升图像质量和细节。CARAFE结合了上下文感知机制和聚合特征的能力,通过动态的上下文注意力机制来提升细节恢复的效果。 理论介绍 传统的卷积操作通常依赖于局部区域…

大型制造企业IT蓝图、信息化系统技术架构规划与实施路线方案

关注 获取ppt​​​​​​全文&#xff0c;请关注作者

HTTP 长连接(HTTP Persistent Connection)简介

HTTP长连接怎么看&#xff1f; HTTP 长连接&#xff08;HTTP Persistent Connection&#xff09;简介 HTTP 长连接&#xff08;Persistent Connection&#xff09;是 HTTP/1.1 的一个重要特性&#xff0c;它允许在一个 TCP 连接上发送多个 HTTP 请求和响应&#xff0c;而无需为…

001集—— 创建一个WPF项目 ——WPF应用程序入门 C#

本例为一个WPF应用&#xff08;.NET FrameWork&#xff09;。 首先创建一个项目 双击xaml文件 双击xaml文件进入如下界面&#xff0c;开始编写代码。 效果如下&#xff1a; 付代码&#xff1a; <Window x:Class"WpfDemoFW.MainWindow"xmlns"http://schema…

微信小程序配置less并使用

1.在VScode中下载Less插件 2.在微信小程序中依次点击如下按钮 选择 从已解压的扩展文件夹安装… 3.选中刚在vscode中下载安装的插件文件 如果没有修改过插件的安装目录&#xff0c;一般是在c盘下C:\用户\用户名.vscode\extensions\mrcrowl.easy-less-2.0.2 我的路径是&#xf…

Vue网页屏保

Vue网页屏保 在vue项目中&#xff0c;如果项目长时间未操作需要弹出屏幕保护程序&#xff0c;以下为网页屏保效果&#xff0c;看板内容为连接的资源。 屏保组件 <template><div v-if"isActive" class"screensaver" click"disableScreens…

【SpringBoot】使用IDEA创建SpringBoot项目

1、使用SpringBoot脚手架创建 我们使用SpringBoot的脚手架Spring Initializr创建&#xff0c;如图所示&#xff1a; 2、选择SpringBoot版本 最开始做项目时候&#xff0c;组长说创建一个 springboot 2.5.4 的项目&#xff0c;mysql使用 5.6.X &#xff0c;maven使用是3.6.X…

如何在鸿蒙API9和x86模拟器中使用MQTT

目录 引言 安装MQTT软件包 避免MQTT软件包自动升级 程序的编写 运行测试 结语 引言 虽然我的课主要是OpenHarmony南向开发的&#xff0c;但是结课时有个同学说他在写鸿蒙APP时无法将MQTT库加入到设备中&#xff0c;希望我帮忙看看。由于他没有鸿蒙的真机&#xff0c;只能…

保姆级教程用vite创建vue3项目并初始化添加PrimeVue UI踩坑实录

文章目录 一、什么是PrimeVue二、详细教程1.添加PrimeVue2.配置main.js3.添加自动引入4.配置vite.config.js5.创建测试页面 一、什么是PrimeVue PrimeVue 是一个用于 Vue.js 3.x 开发的一款高质量、广受欢迎的 Web UI 组件库。 官网地址&#xff1a;https://primevue.org/ 二、…

QT的ui界面显示不全问题(适应高分辨率屏幕)

//自动适应高分辨率 QCoreApplication::setAttribute(Qt::AA_EnableHighDpiScaling);一、问题 电脑分辨率高&#xff0c;默认情况下&#xff0c;打开QT的ui界面&#xff0c;显示不全按钮内容 二、解决方案 如果自己的电脑分辨率较高&#xff0c;可以尝试以下方案&#xff1a;自…