DialogFragment——解决PopupWindow中的输入框无法复制粘贴的问题;Android中的两种弹窗PopupWindow和Dialog的区别。

本文主要是介绍DialogFragment——解决PopupWindow中的输入框无法复制粘贴的问题;Android中的两种弹窗PopupWindow和Dialog的区别。,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文旨在介绍Android中的两种弹窗PopupWindow和Dialog的区别。以及关于写代码的一些思考。

其实之前各类弹窗,都是使用PopupWindow来实现的。因为PopupWindow能实现这些需求,虽然有的时候有些麻烦,比如前面的文章有写到PopupWindow去实现蒙版遮罩的问题。它本身是没有蒙版提供的。

我们项目中使用各种方法去给它增加遮盖层。如今看来确实是有些愚蠢的。我不介意说出以前存在的问题,因为发现问题并改进问题才是学习的过程。当然也希望看到这篇文章的小伙伴有自己的想法可以多多沟通,如果也有跟过去的我一样盲目的使用PopupWindow来做需求的同学,可以从这篇文章中学到东西。

这次遇到PopupWindow中的输入框无法复制粘贴,惯例先去网上搜了一下,发现很多人遇到同样的问题,解决方案全部都是换成DialogFragment就好了。得到这个结果后我有点不服气,为啥我PopupWindow复制粘贴就不行,为啥就非得换成DialogFragment。关于为啥PopupWindow复制粘贴不行,我开始各种搜索,也去学习了一下长按复制粘贴的源码,但是感觉自己的智商不是很够,到现在也没弄明白到底为啥不行。如果有知道的小伙伴希望不吝赐教。

通过这个问题我开始思考,为什么我不管什么弹窗类的需求都用PopupWindow来实现,到底它有啥好的?既然它这么好,那dialog存在的意义又是什么呢。其实以前的做法说白了,就是有项目里有了一套没什么问题的类似代码以后,再来相关的需求,不外乎就是复制粘贴去实现。所以就导致了一直用PopupWindow的这个问题。其实在用的时候,尤其是在强行给PopupWindow加入灰色蒙版的时候,我就有个疑问,为什么PopupWindow不自带蒙版,还要自己去实现,而且网上处理这类问题的文章并不多。这次真的想弄明白官方为什么要提供两种弹窗以后,这些疑惑也就想通了。

简单介绍:

Dialog
默认带灰色遮盖层,会覆盖整个屏幕,使dialog下层的内容不可点击,只有先取消dialog才能继续点击其他内容。
所以当你的需求,是需要一个灰色覆盖层的时候,优先考虑使用Dialog。
Dialog可以指定显示在屏幕个各个位置,可以通过设置偏移实现显示在屏幕的某个位置上。
灰色蒙层的透明度可控,展示位置可控,页面布局可控。

AlertDialog
没啥说的,是Dialog的一个子类,就是为了更方便使用而存在的。

DialogFragment
谷歌官网推荐的Dialog实现方式,使用这个可以让Dialog的生命周期可控。
用法可以参考官网对DialogFragment的介绍:https://developer.android.google.cn/guide/topics/ui/dialogs#java

PopupWindow
不带灰色遮盖层,可以显示在屏幕的任意位置,也可以基于某个view的位置显示。
其他能实现的功能基本与dialog一致。所以其实这两个确实是可以互相替代的。

总结,我的理解:
一般来说,我们开发需求的时候很少会使用到系统自带的样式,基本上都需要自定义样式的。在这一点上,Dialog和PopupWindow并没什么区别,都是可以做到的。

1.那么通常当需求中需要实现灰色蒙版的时候,建议使用DialogFragment实现,自带蒙版。
2.当需求的弹窗里面有输入框的时候,建议使用DialogFragment(因为一般正常都需要支持复制粘贴的)
3.如果需求是要求基于某个view的位置去显示,那么建议使用PopupWindow。如果不带蒙版效果,建议使用PopupWindow。

具体情况还是具体分析啦。不过通过这次让我明白一个道理,官方提供的同类View是针对不同情况下存在更合适使用的。虽然很多功能我们通过一套逻辑就能实现,但是还是应该多去思考一下不同的实现方式到底有什么不同,选一个更合适更合理的方法。

下面附一下我学习了DialogFragment后,对它的一个简单基类封装,旨在对同类弹窗的一个封装,减少每个弹窗都要去初始化dialog代码,让子类只需要处理业务逻辑。

/*** 基于DialogFragment实现的dialog基类(目的:基类完成初始化弹窗等一些配置工作,子类做逻辑处理)* 通过实现getLayoutId()返回布局文件,实现自定义布局* 通过实现initView()完成布局初始化。* 通过调用 show(FragmentManager manager, String tag) 方法展示弹窗。* DialogFragment的dismiss处理了fragment的移除操作,因此每次使用时直接New对象即可,不需考虑复用问题(亲试:考虑的话会存在一些问题)。** 可配置,弹窗位置(当居顶部显示的时候,可配置距顶部的百分比setMarginTop)* 可配置是否可取消* 可配置灰色蒙版透明度** Time:2021/2/4* Author:yang.dong*/
public abstract class BaseDialogFragment extends DialogFragment {protected View mLayout;private AlertDialog dialog;@NonNull@Overridepublic Dialog onCreateDialog(@Nullable Bundle savedInstanceState) {if(dialog == null){AlertDialog.Builder builder = new AlertDialog.Builder(getActivity());LayoutInflater inflater = getActivity().getLayoutInflater();mLayout = inflater.inflate(getLayoutId(), null);builder.setView(mLayout);dialog = builder.create();Window dialogWindow = dialog.getWindow();WindowManager.LayoutParams lp = dialogWindow.getAttributes();dialogWindow.setDimAmount(alpha); //设置灰色遮盖层的透明度dialogWindow.setBackgroundDrawableResource(android.R.color.transparent); //dialog会有默认背景,设置透明后,可以完美展示自己的布局dialogWindow.setGravity(gravity); //设置居顶部显示if(gravity == Gravity.TOP){ //距离屏幕顶部Display d = getActivity().getWindowManager().getDefaultDisplay(); // 获取屏幕宽、高用lp.y = (int)(d.getHeight() * marginTop); // 新位置Y坐标}dialogWindow.setAttributes(lp);dialog.setCanceledOnTouchOutside(cancelable);}initView();return dialog;}@Overridepublic void show(FragmentManager manager, String tag) {//防止重复添加fragment崩溃FragmentTransaction transaction = manager.beginTransaction();Fragment fragment = manager.findFragmentByTag(tag);if(fragment != null){transaction.remove(fragment);}transaction.commit();super.show(manager, tag);}private float alpha = 0.3f;private float marginTop = 0.3f; // 距离顶部的高度(当Gravity为top时,生效)private int gravity = Gravity.TOP;private boolean cancelable = false; //点击灰色区域是否可以取消,默认不取消private void setAlpha(float alpha){this.alpha = alpha;}protected void setMarginTop(float marginTop){this.marginTop = marginTop;}protected void setGravity(int gravity){ //Gravity.TOPthis.gravity = gravity;}protected void setCancel(boolean cancelable){ //Gravity.TOPthis.cancelable = cancelable;}protected abstract int getLayoutId();protected abstract void initView();
}

另外在使用的过程中,遇到两个问题:

1.一直觉得在实例化对象的时候(也就是点击按钮出现弹窗的时候),不应该每次都去创建fragment实例,而应该去判一个空,创建过一次以后,再次点击就只去修改UI。之前使用PopupWindow的时候一直采用这种做法。但是这次真的是,去这样处理感觉给自己挖了一个好大的坑。具体现象就是,第二次点击后UI无法正常刷新。
尝试解决了好久,最后我妥协了。还是每次都去new一个新的吧!也去看了一下dismiss方法的源码,dismiss后是会把fragment给移除掉的。并且在Profiler中去观察了我疯狂点击按钮,一直出弹窗的情况下内存变化。确实真的没啥变化。放心去new就好了。(欢迎指教)

2.快速点击导致闪退。其实这个问题我是为了解决第一个问题,在网上找答案的时候,看别人使用这个遇到的坑,说快速连着点两下就会闪退。我试了一下 ,
果然特喵的闪退了。还真的是。。。我想不出来当时的心情。感觉这种问题真的不该我们来处理的。一个方案是别人帮我踩的坑,如上对show()方法的一个重写。重写了以后连续点两次以后确实不闪退了。不过感觉体验也不是特别好,弹窗还是会出现两次,第一个先消失然后出现第二个。所以我又给点击事件加上了防止连续点击的处理。两手准备吧算是~

这次的分享就到这里啦,欢迎指教,大家加油哦

这篇关于DialogFragment——解决PopupWindow中的输入框无法复制粘贴的问题;Android中的两种弹窗PopupWindow和Dialog的区别。的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1071698

相关文章

qt5cored.dll报错怎么解决? 电脑qt5cored.dll文件丢失修复技巧

《qt5cored.dll报错怎么解决?电脑qt5cored.dll文件丢失修复技巧》在进行软件安装或运行程序时,有时会遇到由于找不到qt5core.dll,无法继续执行代码,这个问题可能是由于该文... 遇到qt5cored.dll文件错误时,可能会导致基于 Qt 开发的应用程序无法正常运行或启动。这种错

Before和BeforeClass的区别及说明

《Before和BeforeClass的区别及说明》:本文主要介绍Before和BeforeClass的区别及说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Before和BeforeClass的区别一个简单的例子当运行这个测试类时总结Before和Befor

SpringBoot排查和解决JSON解析错误(400 Bad Request)的方法

《SpringBoot排查和解决JSON解析错误(400BadRequest)的方法》在开发SpringBootRESTfulAPI时,客户端与服务端的数据交互通常使用JSON格式,然而,JSON... 目录问题背景1. 问题描述2. 错误分析解决方案1. 手动重新输入jsON2. 使用工具清理JSON3.

SpringBoot服务获取Pod当前IP的两种方案

《SpringBoot服务获取Pod当前IP的两种方案》在Kubernetes集群中,SpringBoot服务获取Pod当前IP的方案主要有两种,通过环境变量注入或通过Java代码动态获取网络接口IP... 目录方案一:通过 Kubernetes Downward API 注入环境变量原理步骤方案二:通过

MySQL 设置AUTO_INCREMENT 无效的问题解决

《MySQL设置AUTO_INCREMENT无效的问题解决》本文主要介绍了MySQL设置AUTO_INCREMENT无效的问题解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参... 目录快速设置mysql的auto_increment参数一、修改 AUTO_INCREMENT 的值。

关于跨域无效的问题及解决(java后端方案)

《关于跨域无效的问题及解决(java后端方案)》:本文主要介绍关于跨域无效的问题及解决(java后端方案),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录通用后端跨域方法1、@CrossOrigin 注解2、springboot2.0 实现WebMvcConfig

Go语言中泄漏缓冲区的问题解决

《Go语言中泄漏缓冲区的问题解决》缓冲区是一种常见的数据结构,常被用于在不同的并发单元之间传递数据,然而,若缓冲区使用不当,就可能引发泄漏缓冲区问题,本文就来介绍一下问题的解决,感兴趣的可以了解一下... 目录引言泄漏缓冲区的基本概念代码示例:泄漏缓冲区的产生项目场景:Web 服务器中的请求缓冲场景描述代码

Java死锁问题解决方案及示例详解

《Java死锁问题解决方案及示例详解》死锁是指两个或多个线程因争夺资源而相互等待,导致所有线程都无法继续执行的一种状态,本文给大家详细介绍了Java死锁问题解决方案详解及实践样例,需要的朋友可以参考下... 目录1、简述死锁的四个必要条件:2、死锁示例代码3、如何检测死锁?3.1 使用 jstack3.2

解决JSONField、JsonProperty不生效的问题

《解决JSONField、JsonProperty不生效的问题》:本文主要介绍解决JSONField、JsonProperty不生效的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑... 目录jsONField、JsonProperty不生效javascript问题排查总结JSONField

github打不开的问题分析及解决

《github打不开的问题分析及解决》:本文主要介绍github打不开的问题分析及解决,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、找到github.com域名解析的ip地址二、找到github.global.ssl.fastly.net网址解析的ip地址三