Android:实现弹窗效果

Posted zstar-_

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Android:实现弹窗效果相关的知识,希望对你有一定的参考价值。

效果展示

UI设计想让我实现这样一个弹窗效果,点击中部+号,可以出现一个弹窗,同时可供进一步跳转。
先看最后完成的效果。

为了实现这个效果,主要拆解成几个部分:弹窗绘制、弹窗逻辑编写、弹窗动画

弹窗绘制

首先绘制三个弹窗中的按钮形状
button_circle3.xml

<?xml version="1.0" encoding="UTF-8"?>
<shape
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:shape="rectangle">
    <!-- 填充的颜色 -->
    <solid android:color="@color/bar" />
    <corners
        android:radius="25dip" />
</shape>

之后,构建弹窗文件主体
diglog_j.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:background="@drawable/dialog_bk"
    android:orientation="vertical"
    android:layout_marginLeft="50dp"
    android:layout_marginRight="50dp">

    <Button
        android:id="@+id/btn_fabu"
        android:layout_width="180dp"
        android:layout_height="wrap_content"
        android:background="@drawable/button_circle3"
        android:gravity="center"
        android:text="发布商品"
        android:textStyle="bold"
        android:textColor="@color/white"
        android:textSize="22sp"
        android:layout_marginRight="32dp"
        android:layout_marginLeft="32dp"
        android:layout_marginTop="50dp"
        android:paddingLeft="10dp"
        android:paddingRight="10dp"
        android:layout_gravity="center"/>

    <Button
        android:id="@+id/btn_dingzhi"
        android:layout_width="180dp"
        android:layout_height="wrap_content"
        android:background="@drawable/button_circle3"
        android:gravity="center"
        android:text="定制方案"
        android:textStyle="bold"
        android:textColor="@color/white"
        android:textSize="22sp"
        android:layout_marginRight="32dp"
        android:layout_marginLeft="32dp"
        android:layout_marginTop="30dp"
        android:paddingLeft="10dp"
        android:paddingRight="10dp"
        android:layout_gravity="center"/>


    <Button
        android:id="@+id/btn_renling"
        android:layout_width="180dp"
        android:layout_height="wrap_content"
        android:background="@drawable/button_circle3"
        android:gravity="center"
        android:text="认领小羊"
        android:textStyle="bold"
        android:textColor="@color/white"
        android:textSize="22sp"
        android:layout_marginRight="32dp"
        android:layout_marginLeft="32dp"
        android:layout_marginTop="30dp"
        android:paddingLeft="10dp"
        android:paddingRight="10dp"
        android:layout_gravity="center"
        android:layout_marginBottom="50dp"/>
</LinearLayout>

最后绘制弹窗框的形状
dialog_bk.xml

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item>
        <layer-list>
            <!-- SHADOW LAYER -->
            <item android:top="4dp">
                <shape>
                    <solid android:color="@color/black" />
                    <corners android:radius="30dip"/>
                </shape>

            </item>
            <!-- CONTENT LAYER -->
            <item android:top="12dp">
                <shape>
                    <solid android:color="@color/white" />
                    <corners
                        android:topLeftRadius="20dp"
                        android:topRightRadius="20dp" />
                    <stroke
                        android:width="1dp"
                        android:color="@color/me_blue"
                        />
                </shape>

            </item>
        </layer-list>
    </item>
</selector>

这里比较巧妙的是绘制弹窗框的顶部阴影,使用了两层shape,让第一层往下偏移,实现阴影效果。
再勾勒1dp的蓝色边框,使用stroke关键字

至此,弹窗绘制完成。

弹窗逻辑

弹窗逻辑包含两个部分,一个是弹出逻辑,这部分Dialog已经做了足够的封装,调用相关API即可。另一个是弹窗内按钮的监听和跳转。
这两部分统一写在show_dialog函数中。

private void show_dialog()
        dialog = new Dialog(this, R.style.ActionSheetDialogStyle);
        //填充对话框的布局
        inflate = LayoutInflater.from(this).inflate(R.layout.dialog_j, null);
        //获取控件
        Button btn_fabu = inflate.findViewById(R.id.btn_fabu);//取消
        Button btn_dingzhi = inflate.findViewById(R.id.btn_dingzhi);//支付宝支付
        Button btn_renling = inflate.findViewById(R.id.btn_renling);//添加银行卡
        //将布局设置给Dialog
        dialog.setContentView(inflate);
        //获取当前Activity所在的窗体
        Window dialogWindow = dialog.getWindow();
        //设置Dialog从窗体底部弹出
        dialogWindow.setGravity(Gravity.BOTTOM);
        //获得窗体的属性
        WindowManager.LayoutParams lp = dialogWindow.getAttributes();
        lp.y = 0;//设置Dialog距离底部的距离
        //宽度填充当前布局文件宽度
        lp.width = WindowManager.LayoutParams.WRAP_CONTENT;
        //将属性设置给窗体
        dialogWindow.setAttributes(lp);
        dialog.show();//显示对话框

        btn_fabu.setOnClickListener(new View.OnClickListener() 
            @Override
            public void onClick(View v) 
                Intent i = new Intent(MainActivity.this, fabushangpin.class);
                startActivity(i);
            
        );

        btn_dingzhi.setOnClickListener(new View.OnClickListener() 
            @Override
            public void onClick(View v) 
                Intent i = new Intent(MainActivity.this, dingzhi.class);
                startActivity(i);
            
        );

        btn_renling.setOnClickListener(new View.OnClickListener() 
            @Override
            public void onClick(View v) 
                Intent i = new Intent(MainActivity.this, renling.class);
                startActivity(i);
            
        );
    

弹窗动画

弹窗动画包括两部分,一个是弹窗的效果,即后面的背景变暗;另一个是从下到上的弹出效果。
在上面的函数中,引用了 R.style.ActionSheetDialogStyle,该文件就包含弹窗效果。

在themes.xml中添加

  <style name="ActionSheetDialogStyle" parent="@android:style/Theme.Dialog">
        <!-- 背景透明 -->
        <item name="android:windowBackground">@android:color/transparent</item>
        <item name="android:windowContentOverlay">@null</item>
        <!-- 浮于Activity之上 -->
        <item name="android:windowIsFloating">true</item>
        <!-- 边框 -->
        <item name="android:windowFrame">@null</item>
        <!-- Dialog以外的区域模糊效果 -->
        <item name="android:backgroundDimEnabled">true</item>
        <!-- 无标题 -->
        <item name="android:windowNoTitle">true</item>
        <!-- 半透明 -->
        <item name="android:windowIsTranslucent">true</item>
        <!-- Dialog进入及退出动画 -->
        <item name="android:windowAnimationStyle">@style/ActionSheetDialogAnimation</item>
    </style>
    <!-- ActionSheet进出动画 -->
    <style name="ActionSheetDialogAnimation" parent="@android:style/Animation.Dialog">
        <item name="android:windowEnterAnimation">@anim/actionsheet_dialog_in</item>
        <item name="android:windowExitAnimation">@anim/actionsheet_dialog_out</item>
    </style>

进出动画
actionsheet_dialog_in.xml

<?xml version="1.0" encoding="utf-8"?>
<translate xmlns:android="http://schemas.android.com/apk/res/android"
    android:duration="200"
    android:fromYDelta="100%"
    android:toYDelta="0" />

actionsheet_dialog_out.xml

<?xml version="1.0" encoding="utf-8"?>
<translate xmlns:android="http://schemas.android.com/apk/res/android"
    android:duration="200"
    android:fromYDelta="0"
    android:toYDelta="100%" />

至此,效果完成。

参考资料

https://blog.csdn.net/juer2017/article/details/79012028

以上是关于Android:实现弹窗效果的主要内容,如果未能解决你的问题,请参考以下文章

Qt 实现桌面右下角消息弹窗提示

Android 实现气泡布局/弹窗,可控制气泡尖角方向及偏移量

Android 实现气泡布局/弹窗,可控制气泡尖角方向及偏移量

Android使用BottomSheetBehavior 和 BottomSheetDialog实现底部弹窗

Android使用BottomSheetBehavior 和 BottomSheetDialog实现底部弹窗

Android基础控件——PopupWindow模仿ios底部弹窗