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

本文共 3815 字,大约阅读时间需要 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/

    你可能感兴趣的文章
    non linear processor
    查看>>
    Non-final field ‘code‘ in enum StateEnum‘
    查看>>