[译] 5 分钟让 Drawer 在状态栏下可见

博客介绍如何将抽屉式导航栏元素改造成谷歌 Material Design 规范,使其跨越整个屏幕包括状态栏。具体步骤为扩展主题,创建新样式并让 Activity 指定使用;配置 DrawerLayout 控件,设置相关属性,还提及继承 DrawerLayout 时的处理方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

你也许听过谷歌最新的设计理念 Material Design (“质感设计”)规范,可以让你的抽屉式导航栏跨越整个屏幕,包括状态栏,并且让抽屉后的所有控件以灰暗的网格形式可见。

然而,许多应用打开抽屉式导航栏时看来是这样的

这里将示范如何把这些元素改造成上面说到的规范。

扩展你的主题

你可能已经给包含抽屉的 Activity 定义了一个样式。

<style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar">  
    <item name="colorPrimary">@color/colorPrimary</item>
    <item name="colorPrimaryDark">@color/colorPrimaryDark</item>
    <item name="colorAccent">@color/colorAccent</item>
</style>  
复制代码

第一步,创建一个扩展自AppTheme的新Theme

vaules/styles.xml

<style name="AppTheme.NoActionBar">  
    <item name="windowActionBar">false</item>
    <item name="windowNoTitle">true</item>
</style>
复制代码

v21/styles.xml

<style name="AppTheme.NoActionBar">  
    <item name="windowActionBar">false</item>
    <item name="windowNoTitle">true</item>
    <item name="android:windowDrawsSystemBarBackgrounds">true</item>
    <item name="android:statusBarColor">@android:color/transparent</item>
</style>  
复制代码

并且确保你的 Activity 指定使用了这个 theme,比如

<activity  
    android:name="MyDrawerNavActivity"
    android:theme="@style/AppTheme.NoActionBar"
复制代码
配置 DrawerLayout 控件

第二步,到你定义DrawerLayout控件的地方,设置insetForegroundColor (如果你不想控制 ScrimInsetLayout的颜色,你也可以不设置)。并设置好 fitsSystemWindow 属性值

<android.support.v4.widget.DrawerLayout  
    ...
    android:fitsSystemWindows="true"
    app:insetForeground="@color/inset_color"
    >
复制代码

看起来这样

当然,如果一会你想在代码里改变状态栏的颜色或ScrimInsetLayout的颜色,你可以在DrawerLayout中通过setters方法来获取并改变。

drawerLayout.setStatusBarBackgroundColor(ContextCompat.getColor(this, R.color.wierd_green));  

drawerLayout.setScrimColor(ContextCompat.getColor(this, R.color.wierd_transparent_orange));  
复制代码

感谢你的阅读,如果在我分享的内容里,你有更好的方法来实现,那么在评论里更正,感激不尽。

* 以下添加于 6月5, 2016*

如果你继承 DrawerLayout

Android Support 兼容包(AppCompat) 会在 DrawerLayout 里加入一个 android.support.design.internal.ScrimInsetsFrameLayout, 但如果你使用继承自 DrawerLayout 的自定义控件则不会这么做。

如果你继承了DrawerLayout 但是没有加入ScrimInsetsFrameLayout,你需要这么做:

activity_with_drawer_layout.xml

<com.myproject.views.MyDrawerLayout  
    android:id="@+id/drawer_layout"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:fitsSystemWindows="true">

    <FrameLayout
        android:id="@+id/content_frame"
        android:layout_width="match_parent"
        android:layout_height="match_parent"/>

    <fragment
        android:id="@+id/left_drawer"
        android:name="com.myproject.fragments.NavigationFragment"
        android:layout_width="wrap_content"
        android:layout_height="match_parent"
         />

</com.myproject.views.MyDrawerLayout>  
复制代码

在你的抽屉布局文件中加入一个 ScrimInsetsFrameLayout,如:

navigation_fragment_layout.xml

<?xml version="1.0" encoding="utf-8"?>  
<android.support.design.internal.ScrimInsetsFrameLayout  
    ...
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@android:color/transparent"
    android:fitsSystemWindows="true"
    >

    <!--- content of drawer here --->

</android.support.design.internal.ScrimInsetsFrameLayout>复制代码

转载于:https://juejin.im/post/5cd14aff6fb9a0321e16b7cd

package com.example.kucun2; import android.app.ProgressDialog; import android.os.Bundle; import android.os.Looper; import android.util.Log; import android.view.Menu; import android.widget.Toast; import com.example.kucun2.DataPreserver.Data; import com.google.android.material.snackbar.Snackbar; import com.google.android.material.navigation.NavigationView; import androidx.appcompat.app.AppCompatActivity; import androidx.drawerlayout.widget.DrawerLayout; import androidx.navigation.NavController; import androidx.navigation.Navigation; import androidx.navigation.ui.AppBarConfiguration; import androidx.navigation.ui.NavigationUI; import com.example.kucun2.databinding.ActivityMainBinding; public class MainActivity extends AppCompatActivity { private AppBarConfiguration mAppBarConfiguration; private ActivityMainBinding binding; private OnDataLoadListener dataLoadListener; private static String TAG = "MainActivity"; private ProgressDialog progressDialog; // 等待对话框 @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_loading); // 初始加载界面 loadAppData(); // 启动数据加载 } private void initUI() { // 确保在主线程执行 if (Looper.myLooper() != Looper.getMainLooper()) { runOnUiThread(this::initUI); return; } // 设置主布局 binding = ActivityMainBinding.inflate(getLayoutInflater()); setContentView(binding.getRoot()); Log.d(TAG, "initUI: setContentView(binding.getRoot());"); // 设置工具栏 setSupportActionBar(binding.appBarMain.toolbar); // FAB按钮点击事件 binding.appBarMain.fab.setOnClickListener(view -> Snackbar.make(view, "Replace with your own action", Snackbar.LENGTH_LONG) .setAction("Action", null) .setAnchorView(R.id.fab).show() ); // 导航设置 DrawerLayout drawer = binding.drawerLayout; NavigationView navigationView = binding.navView; // 顶部导航配置 mAppBarConfiguration = new AppBarConfiguration.Builder( R.id.nav_home, R.id.nav_kucun, R.id.nav_add_jinhuo, R.id.nav_diandan) .setOpenableLayout(drawer) .build(); // 导航控制器设置 NavController navController = Navigation.findNavController(this, R.id.nav_host_fragment_content_main); NavigationUI.setupActionBarWithNavController(this, navController, mAppBarConfiguration); NavigationUI.setupWithNavController(navigationView, navController); } @Override public boolean onCreateOptionsMenu(Menu menu) { getMenuInflater().inflate(R.menu.main, menu); return true; } @Override public boolean onSupportNavigateUp() { NavController navController = Navigation.findNavController(this, R.id.nav_host_fragment_content_main); return NavigationUI.navigateUp(navController, mAppBarConfiguration) || super.onSupportNavigateUp(); } public interface OnDataLoadListener { void onDataLoaded(); void onDataError(); } public void setOnDataLoadListener(OnDataLoadListener listener) { this.dataLoadListener = listener; } private void loadAppData() { // 创建并显示等待对话框 progressDialog = new ProgressDialog(this); progressDialog.setMessage("数据加载中,请稍候..."); progressDialog.setCancelable(false); // 不可取消 progressDialog.setIndeterminate(true); // 无限循环样式 progressDialog.show(); Log.d(TAG, "loadAppData: 显示等待对话框"); // 根据Data类定义,使用两个Runnable回调 Data.loadAllData(getApplicationContext(), // 成功回调 () -> { runOnUiThread(() -> { if (progressDialog != null && progressDialog.isShowing()) { progressDialog.dismiss(); } initUI(); if (dataLoadListener != null) { dataLoadListener.onDataLoaded(); } Toast.makeText(MainActivity.this, "数据加载成功", Toast.LENGTH_SHORT).show(); Data.startPeriodicSync(MainActivity.this, 5); }); }, // 失败回调 () -> { runOnUiThread(() -> { if (progressDialog != null && progressDialog.isShowing()) { progressDialog.dismiss(); } if (dataLoadListener != null) { dataLoadListener.onDataError(); } Toast.makeText(MainActivity.this, "数据加载失败,请检查网络连接", Toast.LENGTH_LONG).show(); initUI(); // 即使失败也初始化UI }); } ); } @Override protected void onStop() { super.onStop(); if (isFinishing()) { Data.saveAllData(this); } } @Override protected void onDestroy() { super.onDestroy(); // 确保对话框关闭 if (progressDialog != null && progressDialog.isShowing()) { progressDialog.dismiss(); } progressDialog = null; Data.stopPeriodicSync(); // 确保数据保存 if (!isFinishing()) { Data.saveAllData(this); } } }
06-29
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值