博客
关于我
Android:一个通用的底部导航
阅读量:767 次
发布时间:2019-03-24

本文共 3751 字,大约阅读时间需要 12 分钟。

优化后的文章内容

现代应用的底部导航实现方案

在移动应用开发中,底部导航栏作为用户操作的重要组成部分,几乎无处不在。现有的许多底部导航实现方式使得开发者面临繁琐的代码重复工作。为了解决这一问题,我们可以打造一个通用的底部导航解决方案,帮助开发者快速实现个性化的导航功能。

我们的需求

构建一个能够灵活处理任意数量底部导航模块的通用解决方案。无需通过findViewById手动获取导航按钮,点击事件的管理也需自动化处理。

解决方案思路

  • 以 LinearLayout 为外层容器:设定一个最外层的线性布局,作为底部导航的容器。

  • 导航按钮使用 FrameLayout:每个导航按钮使用 FrameLayout 作为容器,以便于统一管理导航按钮的点击状态。

  • 状态管理:点击导航按钮时,统一修改该 FrameLayout 及其所有子控件的状态(启用/禁用),同时也切换其他导航按钮的状态。

  • 减少代码复杂度:通过动态获取导航按钮数量,统一设置点击事件,无需手动为每个导航按钮设置点击处理逻辑。

  • 实现代码

    MainActivity 布局文件

    MainActivity 代码示例

    public class MainActivity extends AppCompatActivity {    @InjectView(R.id.main_bottom_switcher_container)    LinearLayout mainBottomSwitcherContainer;    private List
    fragments = new ArrayList<>(); @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); ButterKnife.inject(this); initFragment(); initMainBottomSwitcher(); } private void initFragment() { fragments.add(new HomeFragment()); fragments.add(new OrderFragment()); fragments.add(new MeFragment()); fragments.add(new MoreFragment()); // 默认打开首页 onClickListener.onClick(mainBottomSwitcherContainer.getChildAt(0)); } private void initMainBottomSwitcher() { int childCount = mainBottomSwitcherContainer.getChildCount(); for (int i = 0; i < childCount; i++) { FrameLayout childAt = (FrameLayout) mainBottomSwitcherContainer.getChildAt(i); childAt.setOnClickListener(onClickListener); } } private View.OnClickListener onClickListener = new View.OnClickListener() { @Override public void onClick(View view) { int index = mainBottomSwitcherContainer.indexOfChild(view); changeUi(index); changeFragment(index); } }; private void changeUi(int index) { int childCount = mainBottomSwitcherContainer.getChildCount(); for (int i = 0; i < childCount; i++) { if (i == index) { setEnable(mainBottomSwitcherContainer.getChildAt(i), false); } else { setEnable(mainBottomSwitcherContainer.getChildAt(i), true); } } } private void changeFragment(int index) { Fragment fragment = fragments.get(index); getSupportFragmentManager() .beginTransaction() .replace(R.id.main_fragment_container, fragment) .commit(); } private void setEnable(View item, boolean b) { if (item instanceof ViewGroup) { ((ViewGroup) item).setEnabled(b); for (int i = 0; i < ((ViewGroup) item).getChildCount(); i++) { View child = ((ViewGroup) item).findViewByIdAt(i); if (child == null) continue; setEnable(child, b); } } else { item.setEnabled(b); } }}

    求索过程

  • 确定需要通用处理的最小单位:采用 FrameLayout 作为每个导航按钮的容器,这样我们可以统一管理所有导航按钮的状态。

  • 动态获取导航按钮数量:通过 mainBottomSwitcherContainer.getChildCount() 获取到所有导航按钮的数量,并遍历进行处理。

  • 设置统一点击事件:为每个导航按钮动态设置点击事件 listeners,无需手动为每个导航按钮设置 listener。

  • 状态切换逻辑:点击某导航按钮时,切换该导航按钮及其所有子控件的启用状态,同时切换其他导航按钮的状态。

  • �益

    这种通用底部导航解决方案具有以下优势:

    • 灵活性:无论底部导航有多少个模块,都能够轻松添加。

    • 代码简洁:通过动态获取导航按钮数量和统一设置点击事件,降低了代码的复杂性。

    • 维护方便:只需复制导航按钮的布局,添加 Fragment 即可,无需手动修改代码。

    • 性能优化:通过递归方式设置控件状态,保证了所有层级的状态都能正确切换。

    运用示例

  • 添加新导航按钮:复制现有导航按钮的布局,修改文字和图片,添加新的 Fragment 即可。

  • 自定义导航按钮样式:通过修改 FrameLayout 及其子控件的属性,实现个性化的导航按钮风格。

  • 状态切换:在 setEnable 方法中,通过传递 b 参数控制导航按钮的启用状态,实现状态切换的逻辑。

  • 这个通用底部导航解决方案不仅提高了开发效率,还为后续的维护和扩展提供了极大的便利。

    转载地址:http://hxikk.baihongyu.com/

    你可能感兴趣的文章
    NMF(非负矩阵分解)
    查看>>
    nmon_x86_64_centos7工具如何使用
    查看>>
    NN&DL4.1 Deep L-layer neural network简介
    查看>>
    NN&DL4.3 Getting your matrix dimensions right
    查看>>
    NN&DL4.7 Parameters vs Hyperparameters
    查看>>
    NN&DL4.8 What does this have to do with the brain?
    查看>>
    nnU-Net 终极指南
    查看>>
    No 'Access-Control-Allow-Origin' header is present on the requested resource.
    查看>>
    NO 157 去掉禅道访问地址中的zentao
    查看>>
    no available service ‘default‘ found, please make sure registry config corre seata
    查看>>
    No compiler is provided in this environment. Perhaps you are running on a JRE rather than a JDK?
    查看>>
    no connection could be made because the target machine actively refused it.问题解决
    查看>>
    No Datastore Session bound to thread, and configuration does not allow creation of non-transactional
    查看>>
    No fallbackFactory instance of type class com.ruoyi---SpringCloud Alibaba_若依微服务框架改造---工作笔记005
    查看>>
    No Feign Client for loadBalancing defined. Did you forget to include spring-cloud-starter-loadbalanc
    查看>>
    No mapping found for HTTP request with URI [/...] in DispatcherServlet with name ...的解决方法
    查看>>
    No mapping found for HTTP request with URI [/logout.do] in DispatcherServlet with name 'springmvc'
    查看>>
    No module named 'crispy_forms'等使用pycharm开发
    查看>>
    No module named cv2
    查看>>
    No module named tensorboard.main在安装tensorboardX的时候遇到的问题
    查看>>