一、前言
在正常使用 PopupWindow 时不是很难,只需要几行代码就能实现一个对话框,但是有很多方法都是重复的,PopupWindow 本身会有一些局限性,需要处理。在使用 PopupWindows 时,默认是没有阴影背景的,需要自己处理。仅仅使用 PopupWindow 就可以满足项目中大部分的对话框需求。但是 Dialog 也有它的优势,所以可以看具体情况选择合适的对话框。封装后,调用起来更加方便,同时对于一些高频对话框可以做成通用的,以后就不需要再重复去写,随时调用更加方便。
二、引入
[图片上传失败...(image-aea1c3-1597160835850)]
[图片上传失败...(image-51bd6b-1597160749675)]
Add it in your root build.gradle at the end of repositories:
Add it in your root build.gradle at the end of repositories:
allprojects {
repositories {
...
maven { url 'https://www.jitpack.io' }
}
}
Step 2. Add the dependency
dependencies {
implementation 'com.github.KiWiLss:LPopup:1.0.1'
}
三、BasePopup简单使用
BasePopup 是封装的基本对话框,其他对话框可以直接继承 BasePopup 使用,这样使用起来更简单方便。阴影动画效果也很好,PopupWindow 使用总结,这篇文章有对阴影实现方式的分析总结。阴影的实现缺点是部分机型状态栏不能覆盖阴影,整体来说效果还是挺好的。
1. 简单使用示例
- 效果图:
对话框代码:
这里做对话框具体的实现,也可以设置各种想要的效果。
class BasePopupDemo(activity: Activity,layoutId: Int = R.layout.pw_center): BasePopup(activity,layoutId) {
override fun setInterface() {
//这里可以做获取数据,设置数据的操作,点击事件,以及对话框显示前的各种设置都可以在这里
contentView?.run {
tv_pw_onetitle_title.text = "你好"
}
}
}
简单布局:
调用,里面有各种方法的详细注解:
//basepopup 使用示例
btn_common_base.setOnClickListener {
BasePopupDemo(this)
// .setIsMask()//设置是否有阴影
// .setIsClickDismiss()//设置点击外部是否消失
// .setPopupWidth()//设置对话框宽度
// .setPopupHeight()//设置对话框高度
// .setAnimationStyle2()//设置对话框展示动画
.showCenter()
}
2.内置封装对话框使用示例,基本都是偏向 Ios 风格
效果图:- 2.1 一个标题,一个按钮对话框,代码中有详细的注释
Loopopup.Builder(this)
.title("builder 模式")//设置标题
.titleColor(R.color.colorAccent)//标题颜色
.titleIsBold(true)//设置加粗
.outerBg(R.drawable.bg_white_fillet_20)//对话框背景
// .btnText()//按钮文字
// .btnSize()//按钮文字大小
// .btnColor()//按钮文字颜色
// .btnBg()//按钮背景
.clickCallBack(object : LoCallback {
override fun click(loopopup: BasePopup?) {
loopopup?.dismiss()
Toast.makeText(this@CommonPopupActivity, "hello", Toast.LENGTH_SHORT).show()
}
})
.build()
.showCenter()
- 2.2 一个标题,两个按钮样式
Lotpopup.Builder(this)
.rightBg(R.drawable.bg_test_right)//设置按钮背景,圆角要和整个背景圆角大小相同
.callback(object : LtCallback() {
//默认设置了左侧取消按钮,如需要可重写方法,示例如下
override fun right(loopopup: BasePopup?) {
loopopup?.dismiss()
Toast.makeText(this@CommonPopupActivity, "right", Toast.LENGTH_SHORT).show()
}
override fun left(loopopup: BasePopup?) {
super.left(loopopup)
Toast.makeText(this@CommonPopupActivity, "left", Toast.LENGTH_SHORT).show()
}
})
.build()
.showCenter()
- 2.3 两个标题(副标题,可以作为内容),一个按钮
Ltopopup.Builder(this)
.callback(object : LoCallback {
override fun click(loopopup: BasePopup?) {
loopopup?.dismiss()
}
})
.build()
.showCenter()
- 2.4 两个标题(副标题,可以作为内容),两个按钮
val pp = Lttpopup.Builder(this)
.title("任意标题")
.titleColor(R.color.colorAccent)
.titleSize(R.dimen.s15)
.titleIsBold(true)
.subtitle("随意内容")
.subtitleColor(R.color.blue0076)
.subtitleSize(R.dimen.s15)
.leftText("残忍")
.leftSize(R.dimen.s15)
.leftColor(R.color.colorPrimary)
.rightText("立即就去")
.rightSize(R.dimen.s15)
.rightColor(R.color.white)
.rightBg(R.drawable.bg_test_right)
.callback(object : LtCallback() {
override fun right(loopopup: BasePopup?) {
}
})
.build()
//pp.animationStyle = R.style.AnimFadeCenter
pp.showCenter()
三、EasyPopup 使用
EasyPopup 这个对话框,动画效果很好,不会出现阴影无法覆盖的状态栏情况,同时对于位置的控制更好,可以更好的用在菜单型对话框上面。推荐使用这个。
3.1 简单使用示例
效果图:
自定义对话框继承 EasyPopu:
class EasyPopupDemo(activity: Activity,layoutId: Int = R.layout.pw_center): EasyPopup(activity,layoutId) {
override fun setInterface() {
contentView?.run {
//这里可以做获取数据,设置数据的操作,点击事件,以及对话框显示前的各种设置都可以在这里
contentView?.run {
tv_pw_onetitle_title.text = "你好"
}
}
}
}
自定义布局:
调用:
//EasyPopup 使用示例
btn_common_easy.setOnClickListener {
EasyPopupDemo(this)
.setIsMask()//设置是否有阴影
.setIsTouchOutsideDimiss()//设置点击外部是否消失
.setAnimStyle()//设置动画
.showCenter()
}
3.2 底部弹出对话框效果
对话框代码:
class ChoiceHeadPw(activity: Activity,layoutId: Int = R.layout.pw_choice_head): EasyPopup(activity,layoutId) {
override fun setInterface() {
//这里可以初始化界面,设置数据等,设置对话框宽高,动画等都可以
contentView?.run {
tv_pw_choice_head_cancel.setOnClickListener {
dismiss()
}
tv_pw_choice_head_camera.text = "选择相机"
}
}
}
自定义布局:
调用:
//自定义对话框示例,随意一个对话框
btn_common_head2.setOnClickListener {
ChoiceHeadPw(this)
.showBottom()
}
3.3 内置常用对话框
//easypopup 封装使用,每个弹出在中间都加了渐变动画,底部有向上弹出动画,顶部下拉动画,可以自定义动画
-
- 一个标题,一个按钮
对话框使用示例:
Xoopopup.Builder(this)
.title("这是一个标题")//设置标题
.titleColor(R.color.colorPrimary)//设置标题字体颜色
.titleIsBold(true)//设置标题加粗
.titleSize(R.dimen.s15)//设置标题字体大小
//.outerBg()//设置整个对话框背景
.btnText("随意点击")//按钮的文字
.btnSize(R.dimen.s15)//按钮字体大小
.btnColor(R.color.colorAccent)//按钮字体颜色
//.btnBg()//按钮的背景,最好和对话框背景圆角一致
.callback(object : LoCallback2 {
//设置按钮点击
override fun click(loopopup: EasyPopup?) {
loopopup?.dismiss()
}
})
.build()
.showCenter()
-
- 一个标题,两个按钮
使用示例:
Xotpopup.Builder(this)
.title("标题党")//设置标题
.titleColor(R.color.colorAccent)//标题颜色
.titleSize(R.dimen.s12)//标题大小
.titleIsBold(false)//标题是否加粗
//.outerBg()//背景
.leftText("关闭")//左侧按钮文字
.leftColor(R.color.colorAccent)//左侧文字颜色
.leftSize(R.dimen.s15)//左侧文字大小
//.leftBg()//左侧背景
.rightText("是")//右侧文字
//.rightBg()//右侧背景
.rightColor(R.color.blue0076)//右侧文字颜色
.rightSize(R.dimen.s13)//右侧文字大小
.callback(object : LtCallback2() {
//点击回调
override fun right(loopopup: EasyPopup?) {
}
})
.build()
.showCenter()
-
- 两个标题(副标题,可以当内容),一个按钮
使用示例:
Xtopopup.Builder(this)
.title("title")//设置标题
// .titleColor()//设置标题颜色
// .titleSize()//设置标题大小
// .titleIsBold()//设置标题是否加粗
// .subtitleColor()//设置内容颜色
// .subtitle()//设置内容文本
// .subtitleSize()//设置内容字体大小
// .outerBg()//设置背景
// .btnText()//设置点击文本
// .btnBg()//设置点击背景
// .btnColor()//设置点击文字颜色
// .btnSize()//设置点击文字大小
.callback(object : LoCallback2 {
//点击
override fun click(loopopup: EasyPopup?) {
}
}).build()//获取 popupwindow 实例,后面可以调用 popupwindow 的方法
.showCenter()
-
- 两个标题(副标题,可以当内容),两个按钮
使用示例:
Xttpopup.Builder(this)//这里第二个参数可以传你自己的布局(想控件生效,只要对应的控件 id一样就行)
.title("标题加粗")//设置标题
.titleSize(R.dimen.s18)//设置标题字体大小
.titleIsBold(true)//设置标题是否加粗
.titleColor(R.color.colorAccent)//设置标题颜色
.outerBg(R.drawable.bg_white_fillet_20)//设置整个对话框背景
.subtitle("二个标题,二个按钮")//设置副标题,内容
.subtitleSize(R.dimen.s15)//设置副标题内容字体大小
.subtitleColor(R.color.blue0076)//设置副标题字体颜色
.leftText("残忍离开")//设置左侧标题
.leftSize(R.dimen.s12)//设置左侧字体大小
.leftColor(R.color.grayf5f5)//设置左侧字体颜色
//.leftBg(R.drawable.bg_test_right)//设置左侧背景,最好背景左下角圆角和背景圆角一致
.rightText("点开看看")//设置右侧文字
.rightSize(R.dimen.s12)//设置右侧字体大小
.rightColor(R.color.colorAccent)//右侧文字颜色
.rightBg(R.drawable.bg_test_right)//设置右侧背景
.callback(object : LtCallback2() {
//可以选择是否需要取消点击
override fun right(loopopup: EasyPopup?) {//右侧按钮点击
loopopup?.dismiss()
}
override fun left(loopopup: EasyPopup?) {//左侧按钮点击
super.left(loopopup)//这个方法注释可以取消 dimiss
}
})
.build()
.setIsMask(true)//设置是否有阴影
.showCenter()
-
- 选择拍照和打开相册对话框
使用示例:
val xChoiceHead = XChoiceHead(this, object : XChoiceHead.Callback {
override fun camera(xChoiceHead: XChoiceHead) {
}
override fun album(xChoiceHead: XChoiceHead) {
}
})
//.showBottom()
//对标题不满意,可以通过这个方法对内容进行修改
// val tvCamrea = xChoiceHead.getView(R.id.tv_pw_choice_header_take) as TextView?
// tvCamrea?.text = "是否拍照"
xChoiceHead.showBottom()
3.4 菜单形式对话框
这类对话框,原生的 PopupWindow 控制位置不是很好控制,使用原生的,显示在对应的空间下方比较容易实现,但是显示空间的上下左右的位置就有点麻烦了。
对话框代码:
class EasyMenuDemo(activity: Activity,layoutId:Int = R.layout.pw_menu2): EasyPopup(activity,layoutId) {
override fun setInterface() {
}
}
简单布局:
调用:
EasyMenuDemo(this)
.setIsMask(true)//设置是否有阴影
//.setIsTouchOutsideDimiss()//设置点击外部是否消失
.setAnimStyle(R.style.PopDownRightMenu)//设置动画
.showAsDropDown(btn_common_menu)
3.5 控制显示位置
任意简单对话框:
class GravityMenu(activity: Activity,layoutId: Int = R.layout.pw_menu2): EasyPopup(activity,layoutId) {
override fun setInterface() {
}
}
使用示例:
GravityMenu(this)
.showAtAnchorView(btn_common_gravity,VerticalPosition.CENTER,HorizontalPosition.CENTER)
调用时,第一个参数是要显示的对话框,相对位置的控件,第二个参数是垂直方向上的位置,第三个参数是水平方向上的位置,后面还有两个参数是相对 x 和 y 方向的数值,一般这几个参数随意组合就可以实现在想要的控件的 中间、上方、下方、左侧、右侧,其他更复杂的位置配合最后两个参数调整。
3.6 下拉对话框
对话框:
class PullPw(activity: Activity, layoutId: Int = R.layout.pw_pull): EasyPopup(activity,layoutId) {
override fun setInterface() {
//设置宽高,默认是只使用,要修改成下面这样
width = ViewGroup.LayoutParams.MATCH_PARENT
height = ViewGroup.LayoutParams.MATCH_PARENT
contentView?.run {
ll_pw_pull_outer.setOnClickListener {
dismiss()
}
ll_pw_pull_outer2.setOnClickListener {
dismiss()
}
}
}
}
布局:
调用:
//下拉对话框
btn_common_pull.setOnClickListener {
PullPw(this)
.setAnimStyle(R.style.GrowFromTop)//设置动画
.setIsMask(false)//设置没有阴影
.setIsTouchOutsideDimiss(false)//设置点击外部不能消失
.showAsDropDown(btn_common_pull)//显示在这个控件的底部
}
四、Xpopup 简单使用
效果图:上面实现对话框需要继承 EasyPopup,如果对话框不是很复杂的话,没有必须去新建一个类。使用 Xpopup,可以不用继承任何类,直接调用就可以实现对话框效果。内置 Xpopup 类,可以直接使用,可以用于对话框,菜单等.
对话框实现:
val xpopup = Xpopup.Builder(this, R.layout.pw_center)
.alpha(0.6f)//设置背景透明度
.isMask(true)//设置是否显示阴影
.isCancelable(true)//设置点击外部是否消失
.build()//获取Xpopup实例
xpopup.setText(R.id.tv_pw_onetitle_title, "测试")//设置文字
.setOnClick(R.id.tv_pw_onetitle_cancel, View.OnClickListener {
xpopup.dismiss()
}) //设置点击事件
.setPopupWidth(ViewGroup.LayoutParams.WRAP_CONTENT)//设置宽度
.setPopupHeight(ViewGroup.LayoutParams.WRAP_CONTENT)//设置高度
.show()//默认渐变动画,默认显示在中间,可以在这里设置弹出位置和动画
//.setImageResource()//设置资源图片
//.showCenter()//弹出在中间,默认渐变动画
//.showBottom()//底部弹出,默认上推动画
//.showTop()//顶部弹出,默认下拉动画
// .setIsMask(true)//设置是否有阴影
// .setIsTouchOutsideDimiss(true)//设置点击外部是否消失
// .setBackgroundAlpha(0.5f)//设置阴影渐变度
// .setAnimStyle(R.style.AnimFadeCenter)//设置动画效果
//xpopup.getView(R.id.tv_pw_onetitle_title)//通过这个方法获取对应的控件
菜单:
Xpopup.Builder(this, R.layout.pw_menu2)
.build()
.showAsDropDown(btn_common_xpopupMenu)
任意位置:
Xpopup.Builder(this, R.layout.pw_menu2)
.build()
.showAtAnchorView(
btn_common_xpopupAny,
VerticalPosition.BELOW,
HorizontalPosition.ALIGN_RIGHT
)
五、LDialog 简单使用
使用效果图:这个类是对常见的 Dialog 的简单封装,适用于各种居中和底部对话框,需要继承这个类使用。同时可以设置外部点击不能消失,还可以设置点击返回键不能消失。
- 默认样式,什么参数都不设置
Dialog 类:
class DefaultDialog(context: Context) : LDialog(context) {
override fun initInterface(savedInstanceState: Bundle?) {
}
override fun initContentView(): Int = R.layout.pw_center
}
显示:
val dialog = DefaultDialog(this)
dialog.show()
- 设置各类参数
Dialog 类,详细参数都在下面的注释中:
class OneDialog(context: Context) : LDialog(context) {
override fun initInterface(savedInstanceState: Bundle?) {
setWidth(ViewGroup.LayoutParams.MATCH_PARENT) //设置宽度
.setHeight(ViewGroup.LayoutParams.WRAP_CONTENT) //设置高度
.setGravity(Gravity.BOTTOM) //设置显示位置
//.setImageRes()//设置图标
.setAnimationStyle(R.style.PushInBottom)//设置动画
.setText(R.id.tv_pw_choice_head_camera, "测试LDialog") //设置控件文字
.setOnClick(R.id.tv_pw_choice_head_cancel, View.OnClickListener { //设置点击
dismiss()
})
context
val tvCancel = getView(R.id.tv_pw_choice_head_cancel)// 获取控件
//setCancelable(false)//单独设置 false ,dialog弹出后,点击屏幕或物理返回键,dialog不消失
//setCanceledOnTouchOutside(false)// 单独设置 false ,dialog弹出后,点击屏幕dialog不消失,物理返回键消失
setOnDismissListener { //对话框消失监听
Log.e("MMM", ": onDismiss");
}
}
override fun initContentView(): Int = R.layout.pw_choice_head
}
显示:
OneDialog(this)
.show()
六、XDialog 简单使用
效果图:这个类和 LDialog类似,只是这个类使用时不需要继承,直接使用就行。使用于展示比较简单的,交互比较少的对话框。
默认简单使用:
//什么都不设置,默认居中显示,无动画效果
XDialog.Builder(this,R.layout.pw_center)
.build()
.show()
设置参数:
val dialog = XDialog.Builder(this,R.layout.pw_choice_head)
.width(ViewGroup.LayoutParams.MATCH_PARENT)//设置宽度
.height(ViewGroup.LayoutParams.WRAP_CONTENT)//设置高度
.animationStyle(R.style.PushInBottom)//设置动画
.gravity(Gravity.BOTTOM)//设置位置
.build()
dialog.setText(R.id.tv_pw_choice_head_camera,"不想拍照")//设置文字
//.setImageRes()//设置图标
.setOnClick(R.id.tv_pw_choice_head_cancel,View.OnClickListener { //设置点击
dialog.dismiss()
})
.show()
//可以通过getView获取控件
//dialog.getView(R.id.tv_pw_choice_head_cancel)//
七、LDialogFg 简单使用
效果图:这个是对普通的 DialogFragment 的简单封装,相对于 Dialog 和 PopupWindow,DialogFragment 的优势是生命周期和 Fragment 一样,并且在 Activity横竖屏切换时不会新建。
默认样式:
class DefaultDialogFg: LDialogFg() {
override fun initInterface() {
}
override fun initLayoutId(): Int = R.layout.pw_center
}
//LDialogFg使用示例
btn_dialog_lfg.setOnClickListener {
DefaultDialogFg().show(supportFragmentManager,"")
}
设置参数:
class SetDialogFg: LDialogFg() {
override fun initInterface() {
setGravity(Gravity.BOTTOM)//设置位置
setWidth(ViewGroup.LayoutParams.MATCH_PARENT)//设置宽度
setHeight(ViewGroup.LayoutParams.WRAP_CONTENT)//设置高度
setmAnimationStyle(R.style.PushInBottom)//设置动画
// setText()//设置文字
// setImageRes()//设置图标
//设置点击
// setOnClick(R.id.tv_pw_choice_head_cancel,View.OnClickListener {
// dismiss()
// })
//也可以这样设置
tv_pw_choice_head_cancel.setOnClickListener {
dismiss()
}
isCancelable = false//设置点击外部是否消失(设置为false后,点击外部和返回键均不会消失)
}
override fun initLayoutId(): Int = R.layout.pw_choice_head
}
btn_dialog_lfgset.setOnClickListener {
SetDialogFg()
.show(supportFragmentManager,"")
}
八、地址
LPopup
参考:EasyPopup(已停止维护
Android基于DialogFragment封装一个通用的Dialog-阿里云开发者社区
让你的Dialog变得更简洁一点吧 - 掘金
变种 Builder 模式:优雅的对象构建方式张拭心的博客 shixinzhang-CSDN博客
弹出PopupWindow后让背景变暗的方法
【Android】在任意位置弹出PopupWindow
Gavin-ZYX/SmartPopupWindow