IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 移动开发 -> Android底部弹窗实现方案 -> 正文阅读

[移动开发]Android底部弹窗实现方案

项目中经常会遇到底部弹窗,例如分享弹窗等,今天就来把底部弹窗实现方案总结一下。

BottomSheetDialog

底部表单样式的对话框基类。依赖于Behavior机制。

依赖

dependencies {
    implementation 'com.google.android.material:material:1.4.0'
}

基类封装

public abstract class BaseBottomSheetDialog extends BottomSheetDialog {

    public BaseBottomSheetDialog(@NonNull Context context) {
        super(context, R.style.Theme_BottomSheetDialog_Base);
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        setContentView(getLayoutResId());

        initView();
    }

    @Override
    protected void onStart() {
        super.onStart();

        changePeekHeight();

        Window window = getWindow();
        if (window != null) {
            window.setLayout(WindowManager.LayoutParams.MATCH_PARENT, getPeekHeight());
            window.setGravity(Gravity.BOTTOM | Gravity.CENTER_HORIZONTAL);
        }
    }

    private void changePeekHeight() {
        final int peekHeight = getPeekHeight();
        if (peekHeight > 0) {
            View bottomSheet = findViewById(com.google.android.material.R.id.design_bottom_sheet);
            BottomSheetBehavior<View> behavior = BottomSheetBehavior.from(bottomSheet);
            behavior.setPeekHeight(peekHeight);
            ViewGroup.LayoutParams layoutParams = bottomSheet.getLayoutParams();
            layoutParams.height = peekHeight;
            bottomSheet.setLayoutParams(layoutParams);
        }
    }

    protected abstract int getLayoutResId();

    protected abstract void initView();

    protected int getPeekHeight() {
        return WindowManager.LayoutParams.WRAP_CONTENT;
    }
}

为了去掉BottomSheetDialog默认的背景,需要修改主题样式:

 <style name="Theme.BottomSheetDialog.Base" parent="Theme.Design.Light.BottomSheetDialog">
     <item name="bottomSheetStyle">@style/BottomSheetDialogStyle</item>
 </style>

 <style name="BottomSheetDialogStyle" parent="Widget.Design.BottomSheet.Modal">
     <item name="android:background">@android:color/transparent</item>
 </style>

除此之外,如果需要对生命周期进行感知或者自动恢复,还可以使用BottomSheetDialogFragment作为继承基类。

自定义Dialog

继承Dialog实现自定义弹窗。

public abstract class BaseBottomDialog extends AppCompatDialog {

    public BaseBottomDialog(Context context) {
        super(context, R.style.Theme_BottomDialog_Base);
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        setContentView(getLayoutResId());

        initView();
    }

    @Override
    protected void onStart() {
        super.onStart();

        Window window = getWindow();
        if (window != null) {
            window.setLayout(WindowManager.LayoutParams.MATCH_PARENT, getPeekHeight());
            window.setGravity(Gravity.BOTTOM | Gravity.CENTER_HORIZONTAL);
        }
    }

    protected abstract int getLayoutResId();

    protected abstract void initView();

    protected int getPeekHeight() {
        return WindowManager.LayoutParams.WRAP_CONTENT;
    }
}

为了去掉弹窗自带的背景,需要自定义主题样式:

 <style name="Theme.BottomDialog.Base" parent="android:style/Theme.Dialog">
        <item name="android:windowBackground">@android:color/transparent</item>
        <item name="android:windowNoTitle">true</item>
        <item name="android:windowIsFloating">true</item>
        <item name="android:windowContentOverlay">@null</item>
        <item name="android:windowAnimationStyle">@style/Animation.BottomDialog</item>
    </style>

    <style name="Animation.BottomDialog" parent="Animation.AppCompat.Dialog">
        <item name="android:windowEnterAnimation">@anim/design_bottom_sheet_slide_in</item>
        <item name="android:windowExitAnimation">@anim/design_bottom_sheet_slide_out</item>
    </style>

其中的弹窗动画样式为:

design_bottom_sheet_slide_in

<set xmlns:android="http://schemas.android.com/apk/res/android"
    android:duration="@integer/bottom_sheet_slide_duration"
    android:interpolator="@android:anim/accelerate_decelerate_interpolator">

  <translate
      android:fromYDelta="20%p"
      android:toYDelta="0"/>

  <alpha
      android:fromAlpha="0.0"
      android:toAlpha="1.0"/>

</set>

design_bottom_sheet_slide_out

<set xmlns:android="http://schemas.android.com/apk/res/android"
    android:duration="@integer/bottom_sheet_slide_duration"
    android:interpolator="@android:anim/accelerate_interpolator">

  <translate
      android:fromYDelta="0"
      android:toYDelta="20%p"/>

  <alpha
      android:fromAlpha="1.0"
      android:toAlpha="0.0"/>

</set>

WindowManager

通过WindowManager添加视图到窗口的方式。

public abstract class BaseBottomPanel {

    protected Context context;

    private WindowManager mWindowManager;
    private WindowManager.LayoutParams mParams;

    private boolean mShowing = false;

    private View mView;

    public BaseBottomPanel(Context context) {
        this.context = context;
        mWindowManager = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
    }

    public void show() {
        if (!mShowing) {
            inflate();
            try {
                mWindowManager.addView(mView, mParams);
                mShowing = true;
            } catch (Exception e) {
                mShowing = false;
                return;
            }
        }

        initView();
    }

    public void dismiss() {
        if (mShowing) {
            try {
                mWindowManager.removeView(mView);
                mShowing = false;
            } catch (Exception e) {
                mShowing = true;
            }
        }
    }

    private void inflate() {
        mView = LayoutInflater.from(context).inflate(getLayoutResId(), null);

        mParams = new WindowManager.LayoutParams();
        mParams.gravity = Gravity.BOTTOM | Gravity.CENTER_HORIZONTAL;
        mParams.format = PixelFormat.TRANSLUCENT;
        mParams.type = WindowManager.LayoutParams.TYPE_APPLICATION_ATTACHED_DIALOG;
        mParams.flags = WindowManager.LayoutParams.FLAG_DIM_BEHIND
                | WindowManager.LayoutParams.FLAG_LAYOUT_IN_SCREEN
                | WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS;
        mParams.windowAnimations = R.style.Animation_BottomDialog;
        mParams.width = WindowManager.LayoutParams.MATCH_PARENT;
        mParams.height = getPeekHeight();
        mParams.dimAmount = 0.6f;
    }

    public Context getContext() {
        return context;
    }

    protected <T extends View> T findViewById(@IdRes int id) {
        if (mView == null) {
            return null;
        }
        return mView.findViewById(id);
    }

    protected abstract int getLayoutResId();

    protected abstract void initView();

    protected int getPeekHeight() {
        return WindowManager.LayoutParams.WRAP_CONTENT;
    }
}

窗口动画为:

 <style name="Animation.BottomDialog" parent="Animation.AppCompat.Dialog">
     <item name="android:windowEnterAnimation">@anim/design_bottom_sheet_slide_in</item>
     <item name="android:windowExitAnimation">@anim/design_bottom_sheet_slide_out</item>
 </style>

PopupWindow

通过PopWindow指定Gravity为Bottom来实现底部弹窗、

PopupWindow popupWindow = new PopupWindow();
View view = LayoutInflater.from(this).inflate(R.layout.dialog_share, null);
popupWindow.setContentView(view);
popupWindow.setWidth(WindowManager.LayoutParams.MATCH_PARENT);
popupWindow.setHeight(WindowManager.LayoutParams.WRAP_CONTENT);
popupWindow.setBackgroundDrawable(new ColorDrawable(Color.TRANSPARENT));
popupWindow.showAtLocation(findViewById(android.R.id.content), Gravity.BOTTOM, 0, 0);

以上只是个人总结的部分基类封装或实现,有需要的可以自行拷贝并进行修改以契合个人需求,同时也希望能给个赞,谢谢。

感谢大家的支持,如有错误请指正,如需转载请标明原文出处!

  移动开发 最新文章
Vue3装载axios和element-ui
android adb cmd
【xcode】Xcode常用快捷键与技巧
Android开发中的线程池使用
Java 和 Android 的 Base64
Android 测试文字编码格式
微信小程序支付
安卓权限记录
知乎之自动养号
【Android Jetpack】DataStore
上一篇文章      下一篇文章      查看所有文章
加:2022-03-21 21:02:21  更:2022-03-21 21:02:38 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/24 18:55:49-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码