Android 底部tab,使用recycleview实现

news/2025/1/2 8:32:16/

res/layout/activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:app="http://schemas.android.com/apk/res-auto"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"tools:context=".MainActivity"><!-- 用于显示Fragment的容器 --><FrameLayoutandroid:id="@+id/fragment_container"android:layout_width="0dp"android:layout_height="0dp"app:layout_constraintBottom_toTopOf="@id/rv_navigation"app:layout_constraintLeft_toLeftOf="parent"app:layout_constraintRight_toRightOf="parent"app:layout_constraintTop_toTopOf="parent" /><!-- 底部导航栏 --><androidx.recyclerview.widget.RecyclerViewandroid:id="@+id/rv_navigation"android:layout_width="match_parent"android:layout_height="wrap_content"android:background="@color/white"android:overScrollMode="never"app:layout_constraintBottom_toBottomOf="parent"app:layout_constraintLeft_toLeftOf="parent"app:layout_constraintRight_toRightOf="parent" /></androidx.constraintlayout.widget.ConstraintLayout>

java/com/zs/test/MainActivity.java

package com.zs.test;import android.os.Bundle;
import android.os.Handler;
import android.os.Looper;
import android.view.View;
import android.widget.Toast;import com.tencent.mm.opensdk.openapi.IWXAPI;
import com.tencent.mm.opensdk.openapi.WXAPIFactory;
import com.zs.test.bean.MainNavigation;
import com.zs.test.common.Global;
import com.zs.test.event.Event;
import com.zs.test.event.EventBus;
import com.zs.test.event.Subscribe;
import com.zs.test.fragment.MyFragment;
import com.zs.test.fragment.TestFragment;
import com.zs.test.fragment.WebViewFragment2;
import com.zs.test.ui.adapter.MainNavigationAdapter;import java.util.List;import androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity;
import androidx.fragment.app.Fragment;
import androidx.recyclerview.widget.RecyclerView;public class MainActivity extends AppCompatActivity {private boolean isBackPressedOnce = false; // 标志用户是否已经按过一次返回键private final Handler handler = new Handler(Looper.getMainLooper()); // 用于延时清除标志private WebViewFragment2 webViewFragment2;public static IWXAPI api;private RecyclerView mRvNavigation;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);EventBus.register(this);mRvNavigation = findViewById(R.id.rv_navigation);final MainNavigationAdapter adapter = new MainNavigationAdapter();adapter.setData(MainNavigation.obtainMainNav());adapter.attachToRecyclerView(mRvNavigation);adapter.setOnCheckedChangeListener(this::onCheckedChanged);//设置默认选中adapter.select(0);api = WXAPIFactory.createWXAPI(getApplicationContext(), Global.vxAppId, false);api.registerApp(Global.vxAppId);}private final Fragment[] mFragments = new Fragment[]{new TestFragment(), new MyFragment()};public void onCheckedChanged(@NonNull MainNavigationAdapter adapter, int position) {final String tagStart = "main_fragment_tag_";final List<Fragment> fragments = getSupportFragmentManager().getFragments();for (Fragment f : fragments) {if (f.getTag() != null && f.getTag().startsWith(tagStart)) {getSupportFragmentManager().beginTransaction().hide(f).commitAllowingStateLoss();}}final String tag = tagStart + position;Fragment current = getSupportFragmentManager().findFragmentByTag(tag);if (current != null) {getSupportFragmentManager().beginTransaction().show(current).commitAllowingStateLoss();} else {current = mFragments[position];getSupportFragmentManager().beginTransaction().add(R.id.fragment_container, current, tag).commitAllowingStateLoss();}}private void handleBottomShow(boolean show) {if (show) {runOnUiThread(() -> {if (mRvNavigation != null) mRvNavigation.setVisibility(View.VISIBLE);});} else {runOnUiThread(() -> {if (mRvNavigation != null) mRvNavigation.setVisibility(View.GONE);});}}@Overridepublic void onBackPressed() {// 检查当前显示的 Fragmentfinal List<Fragment> fragments = getSupportFragmentManager().getFragments();for (Fragment fragment : fragments) {if (fragment instanceof TestFragment) {((TestFragment) fragment).handleWebViewBackPress();return;}if (fragment instanceof MyFragment) {((MyFragment) fragment).handleWebViewBackPress();return;}}if (isBackPressedOnce) {// 第二次按返回键,退出应用super.onBackPressed();finishAffinity(); // 结束所有活动,退出应用System.exit(0); // 彻底退出进程} else {// 第一次按返回键,提示用户isBackPressedOnce = true;Toast.makeText(this, "再按一次返回键退出程序", Toast.LENGTH_SHORT).show();// 延时2秒后重置标志handler.postDelayed(() -> isBackPressedOnce = false, 2000);}}@Overrideprotected void onDestroy() {super.onDestroy();// 清除所有回调,避免内存泄漏handler.removeCallbacksAndMessages(null);EventBus.unregister(this);}// 处理 RunGoalEvent 事件@Subscribepublic void MainBottomShow(Event.MainBottomShow event) {handleBottomShow(event.isShow());}
}

res/menu/menu_main.xml

<?xml version="1.0" encoding="utf-8"?>
<menu xmlns:android="http://schemas.android.com/apk/res/android"xmlns:app="http://schemas.android.com/apk/res-auto"><itemandroid:id="@+id/tab_test"android:icon="@drawable/tab_statistics_selector"android:title="趣测"/><itemandroid:id="@+id/tab_my"android:icon="@drawable/tab_step_count_selector"android:title="我的"/></menu>

java/com/zs/test/base/BaseAdapter.java

package com.zs.test.base;import androidx.recyclerview.widget.RecyclerView;/*** {@link RecyclerView}适配器基类*/
public abstract class BaseAdapter extends RecyclerView.Adapter<BaseHolder> {
}

java/com/zs/test/base/BaseHolder.java

package com.zs.test.base;import android.content.Context;
import android.graphics.Color;
import android.graphics.drawable.Drawable;
import android.graphics.drawable.GradientDrawable;
import android.util.SparseArray;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;
import android.widget.TextView;import androidx.annotation.ColorRes;
import androidx.annotation.DrawableRes;
import androidx.annotation.IdRes;
import androidx.annotation.LayoutRes;
import androidx.annotation.NonNull;
import androidx.collection.ArrayMap;
import androidx.core.content.ContextCompat;
import androidx.recyclerview.widget.RecyclerView;/*** RecyclerView ViewHolder*/
public class BaseHolder extends RecyclerView.ViewHolder {private final Context mContext;private final SparseArray<View> mSparseArray;public BaseHolder(@NonNull View itemView) {super(itemView);mContext = itemView.getContext();mSparseArray = new SparseArray<>();}public BaseHolder(@LayoutRes int resource, @NonNull ViewGroup parent) {this(LayoutInflater.from(parent.getContext()).inflate(resource, parent, false));}/*** 设置 itemView最小高度*/public BaseHolder setMinHeight(int minHeight) {itemView.setMinimumHeight(minHeight);return this;}public Context getContext() {return mContext;}@SuppressWarnings("all")public <T extends View> T findViewById(@IdRes int id) {View view = mSparseArray.get(id);if (view == null) {view = itemView.findViewById(id);mSparseArray.put(id, view);}return ((T) view);}public <T extends View> T findViewById(@IdRes int id, boolean visibility) {T view = findViewById(id);view.setVisibility(visibility ? View.VISIBLE : View.GONE);return view;}public <T extends View> T findViewById(@IdRes int id, int visibility) {T view = findViewById(id);view.setVisibility(visibility);return view;}public void setText(@IdRes int id, CharSequence text) {TextView textView = findViewById(id);textView.setText(text);}public void setText(@IdRes int id, int textResOrNum) {TextView textView = findViewById(id);try {textView.setText(getContext().getString(textResOrNum));} catch (Exception e) {textView.setText(String.valueOf(textResOrNum));}}public void setHint(@IdRes int id, CharSequence text) {TextView textView = findViewById(id);textView.setHint(text);}public void setHint(@IdRes int id, int textResOrNum) {TextView textView = findViewById(id);try {textView.setHint(getContext().getString(textResOrNum));} catch (Exception e) {textView.setHint(String.valueOf(textResOrNum));}}public ImageView findImageById(@IdRes int imageViewId) {return findViewById(imageViewId);}public ImageView findImageById(@IdRes int imageViewId, boolean visibility) {return findViewById(imageViewId, visibility);}public ImageView findImageById(@IdRes int imageViewId, int visibility) {return findViewById(imageViewId, visibility);}public void setVisibility(int id, int visibility) {findViewById(id).setVisibility(visibility);}public void setVisibility(int id, boolean visibility) {findViewById(id).setVisibility(visibility ? View.VISIBLE : View.GONE);}public void setImageResource(@IdRes int imageViewId, @DrawableRes int drawableRes) {ImageView imageView = findViewById(imageViewId);imageView.setImageResource(drawableRes);}private final ArrayMap<String, Drawable> mCircleDrawableMap = new ArrayMap<>();/*** 获取圆形的Drawable** @param size  宽高* @param color 颜色*/public Drawable getCircleDrawable(int size, int color) {String key = String.valueOf(size) + color;Drawable drawable = mCircleDrawableMap.get(key);if (drawable == null) {GradientDrawable gradientDrawable = new GradientDrawable();gradientDrawable.setShape(GradientDrawable.OVAL);if (size > 0) {gradientDrawable.setSize(size, size);}gradientDrawable.setColor(color);drawable = gradientDrawable;mCircleDrawableMap.put(key, drawable);}return drawable;}public Drawable getCircleDrawable(int size) {return getCircleDrawable(size, Color.parseColor("#EDEDED"));}public int getColor(@ColorRes int id) {return ContextCompat.getColor(getContext(), id);}}

res/layout/item_main_navigation.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="match_parent"android:layout_height="wrap_content"android:gravity="center"android:minHeight="48dp"android:orientation="vertical"><androidx.appcompat.widget.AppCompatImageViewandroid:id="@+id/iv_icon"android:layout_width="wrap_content"android:layout_height="23dp"android:adjustViewBounds="true"android:scaleType="fitXY" /><androidx.appcompat.widget.AppCompatTextViewandroid:id="@+id/tv_name"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_marginTop="3dp"android:textSize="12sp" /></LinearLayout>

java/com/zs/test/bean/MainNavigation.java

package com.zs.test.bean;import androidx.annotation.DrawableRes;import com.zs.test.R;import java.util.Arrays;
import java.util.List;/*** 底部导航数据*/
public class MainNavigation {/*** 获取主页的*/public static List<MainNavigation> obtainMainNav() {return Arrays.asList(new MainNavigation("趣测", R.drawable.ic_statistics_unselected, R.drawable.ic_statistics_selected),new MainNavigation("我的", R.drawable.ic_step_count_unselected, R.drawable.ic_step_count_selected));}/*** 按钮名称*/private final String buttonName;/*** 图标选中图标*/@DrawableResprivate final int drawableNormal, drawableSelect;public MainNavigation(String buttonName, int drawableNormal, int drawableSelect) {this.buttonName = buttonName;this.drawableNormal = drawableNormal;this.drawableSelect = drawableSelect;}public int getDrawableNormal() {return drawableNormal;}public int getDrawableSelect() {return drawableSelect;}public String getButtonName() {return buttonName;}
}

java/com/zs/test/ui/adapter/MainNavigationAdapter.java

package com.zs.test.ui.adapter;import android.graphics.Color;
import android.view.ViewGroup;
import android.widget.TextView;import com.zs.test.R;
import com.zs.test.base.BaseAdapter;
import com.zs.test.base.BaseHolder;
import com.zs.test.bean.MainNavigation;import java.util.ArrayList;
import java.util.List;import androidx.annotation.NonNull;
import androidx.recyclerview.widget.GridLayoutManager;
import androidx.recyclerview.widget.LinearLayoutManager;
import androidx.recyclerview.widget.RecyclerView;/*** 底部导航*/
public class MainNavigationAdapter extends BaseAdapter {private RecyclerView mRecyclerView;/*** 数据*/private final List<MainNavigation> mData = new ArrayList<>();/*** 当前选中的*/private int mSelect = -1;/*** 选中监听*/private OnCheckedChangeListener mOnCheckedChangeListener;public void setData(List<MainNavigation> data) {mData.clear();if (data != null) mData.addAll(data);notifyDataSetChanged();if (mRecyclerView != null) attachToRecyclerView(mRecyclerView);}public void select(int position) {if (mSelect == position) {return;}final int old = mSelect;mSelect = position;if (old >= 0 && old < mData.size()) notifyItemChanged(old);if (mSelect >= 0 && mSelect < mData.size()) notifyItemChanged(mSelect);if (mOnCheckedChangeListener != null && mSelect >= 0) {mOnCheckedChangeListener.onCheckedChanged(this, mSelect);}}public interface OnCheckedChangeListener {void onCheckedChanged(@NonNull MainNavigationAdapter adapter, int position);}/*** 设置选中监听*/public void setOnCheckedChangeListener(OnCheckedChangeListener l) {mOnCheckedChangeListener = l;if (mOnCheckedChangeListener != null && mSelect >= 0) {mOnCheckedChangeListener.onCheckedChanged(this, mSelect);}}@NonNull@Overridepublic BaseHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {return new BaseHolder(R.layout.item_main_navigation, parent);}@Overridepublic void onBindViewHolder(@NonNull BaseHolder holder, int position) {final MainNavigation item = mData.get(position);//按钮图标holder.setImageResource(R.id.iv_icon, position == mSelect ? item.getDrawableSelect() : item.getDrawableNormal());//按钮文本final TextView tvName = holder.findViewById(R.id.tv_name);tvName.setText(item.getButtonName());tvName.setTextColor(Color.parseColor(position == mSelect ? "#574DBE" : "#999999"));//点击事件holder.itemView.setOnClickListener(v -> select(position));}@Overridepublic int getItemCount() {return mData.size();}public void attachToRecyclerView(@NonNull RecyclerView recyclerView) {final int spanCount = Math.max(mData.size(), 1);recyclerView.setLayoutManager(new GridLayoutManager(recyclerView.getContext(), spanCount, GridLayoutManager.VERTICAL, false));recyclerView.setAdapter(this);mRecyclerView = recyclerView;}
}

http://www.ppmy.cn/news/1559548.html

相关文章

排序算法(系列)

希尔排序&#xff08;Shell Sort&#xff09;是一种插入排序的改进版本。它是基于插入排序的思想&#xff0c;通过将待排序的元素进行分组&#xff0c;然后对每组进行插入排序&#xff0c;逐步减少分组的大小&#xff0c;最终完成排序。希尔排序的核心思想是将序列分为多个子序…

基于c语言的union、字符串、格式化输入输出

结构体之共用体union 共用体也叫联合体&#xff0c;其关键词为union 与结构体不同的是&#xff0c;共用体所开辟的存储空间仅仅为其中最长类型变量的存储空间而不是全部变量的存储空间&#xff0c;由于同一内存单元在同一时间内只能存放其中一种的数据类型&#xff0c;因此在每…

评分模型在路网通勤习惯分析中的应用——提出问题(1)

1、问题的由来、目标和意义 最近一段时间和公司其它业务部门讨论时&#xff0c;发现一个有趣的交通路网问题&#xff0c;车辆从S点行驶到V点共用时40分钟&#xff0c;这段时间内路网中的卡口摄像头识别到了车辆通过的信息。如下图所示&#xff1a; 设计师需要通过这些有限的路…

OCR实践-Table-Transformer

前言 书接上文 OCR实践—PaddleOCR Table-Transformer 与 PubTables-1M table-transformer&#xff0c;来自微软&#xff0c;基于Detr&#xff0c;在PubTables1M 数据集上进行训练&#xff0c;模型是在提出数据集同时的工作&#xff0c; paper PubTables-1M: Towards comp…

# 光速上手 - JPA 原生 sql DTO 投影

前言 使用 JPA 时&#xff0c;我们一般通过 Entity 进行实体类映射&#xff0c;从数据库中查询出对象。然而&#xff0c;在实际开发中&#xff0c;有时需要自定义查询结果并将其直接映射到 DTO&#xff0c;而不是实体类。这种需求可以通过 JPA 原生 SQL 查询和 DTO 投影 来实现…

Flink如何处理迟到数据?

在flink中进行窗口计算时&#xff0c;由于乱序流数据的问题&#xff0c;往往会出现迟到数据&#xff0c;迟到数据未参与所属窗口的计算会对计算结果的准确性产生影响&#xff0c;对此&#xff0c;Flink有如下三种方法来保障结果的准确性。 &#xff08;1&#xff09;水位线的延…

Redis可视化工具 RDM mac安装使用

第一步&#xff1a;https://pan.baidu.com/s/10vpdhw7YfDD7G4yZCGtqQg?at1673701651004将dmg下载 第二部&#xff1a;点击下载的dmg文件进行安装、mac可能会提示&#xff1a; 无法验证此App不包含恶意软件 解决方法&#xff1a; 打开系统偏好设置>安全性与隐私>通用&am…

Qt仿音乐播放器:绘画、图片

一、铺垫 1.Qt中给程序员提供的组件&#xff0c;基本上都是矩形&#xff0c;那如果程序员想画一个三角形和圆形&#xff1b;那就必须要使用绘画类&#xff1b; 二、绘画 注意&#xff1a;关于 paintEvent&#xff1a;paintEvent 会在以下情况下被触发: 1.控件⾸次创建. 2.控…