先说说案例 我的项目里有一个搜索功能 是在Activity上直接弹窗 之后选择时间需要弹出一个PickTimeView的时间选择器,这控件个在github上有很多。由此所衍生出了许多问题,让人摸不到头脑。
先看看这个能否帮到你,如果不可以在尝试下面的操作。
mPopWindow.setFocusable(false);
mPopWindow.update();
先上代码
PopupWindow mPopWindow;
TimePickerView timePickerView;
/**
* 初始化popupwindow
*/
private void initPop() {
if (mPopWindow == null) {
View inflate = LayoutInflater.from(getApplicationContext()).inflate(
R.layout.popupwindow_log_doc, null);//
mPopWindow = new PopupWindow(inflate,
ViewPager.LayoutParams.MATCH_PARENT, ViewPager.LayoutParams.WRAP_CONTENT, true);
mPopWindow.setContentView(inflate);
mPopWindow.setBackgroundDrawable(new ColorDrawable());
mPopWindow.setAnimationStyle(R.style.PopupAnimation);
mPopWindow.setFocusable(false);
mPopWindow.setOutsideTouchable(false);
final TextView edit_starttime = inflate.findViewById(R.id.edit_starttime);
final TextView edit_endtime = inflate.findViewById(R.id.edit_endtime);
inflate.findViewById(R.id.rb_disagree).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
start_time = "";
end_time = "";
edit_endtime.setText("");
edit_starttime.setText("");
}
});
inflate.findViewById(R.id.rb_agree).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
if (logRecyclerview != null) {
end_time = edit_endtime.getText().toString().trim();
start_time = edit_starttime.getText().toString().trim();
logRecyclerview.refresh();
}
mPopWindow.dismiss();
}
});
timePickerView = new TimePickerView.Builder(MyLogActivity.this, new TimePickerView.OnTimeSelectListener() {
@Override
public void onTimeSelect(Date date, View v) {
String choosedate = DateUtils.getDate(date.toString());
switch (currentType) {
case R.id.edit_starttime:
edit_starttime.setText(choosedate);
break;
case R.id.edit_endtime:
edit_endtime.setText(choosedate);
break;
}
timePickerView.dismiss();
}
}).setType(new boolean[]{true, true, true, false, false, false}).build();
edit_endtime.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
//选中事件回调
if (!timePickerView.isShowing()) {
timePickerView.show();
currentType = R.id.edit_endtime;
}
}
});
edit_starttime.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
//选中事件回调
if (!timePickerView.isShowing()) {
timePickerView.show();
currentType = R.id.edit_starttime;
}
}
});
mPopWindow.setOnDismissListener(new PopupWindow.OnDismissListener() {
@Override
public void onDismiss() {
Utility.setBackgroundAlpha(MyLogActivity.this, 1f);
}
});
}
Utility.setBackgroundAlpha(MyLogActivity.this, 0.9f);
mPopWindow.showAsDropDown(titelRl);
}
@Override
public void onBackPressed() {
if (mPopWindow != null && mPopWindow.isShowing()) {
mPopWindow.dismiss();
} else super.onBackPressed();
}
@Override
public boolean onTouchEvent(MotionEvent event) {
if (mPopWindow != null && mPopWindow.isShowing()) {
mPopWindow.dismiss();
return false;
} else
return super.onTouchEvent(event);
}
问题一:时间选择器弹出后可能被前一个弹窗覆盖,即在搜索窗之下。
为避免该问题 高度使其为 WRAP_CONTENT
new PopupWindow(inflate,
ViewPager.LayoutParams.MATCH_PARENT, ViewPager.LayoutParams.WRAP_CONTENT, true);
问题二:搜索窗在时间选择器之前关闭,即点返回键或者或者触摸其他位置时搜索窗先与时间选择器关闭。
//重写Activity的onTouchEvent 方法
@Override
public boolean onTouchEvent(MotionEvent event) {
if (mPopWindow != null && mPopWindow.isShowing()) {
mPopWindow.dismiss();
return false;
} else
return super.onTouchEvent(event);
}
//不获取焦点
mPopWindow.setFocusable(false);
//重写返回键
@Override
public void onBackPressed() {
if (mPopWindow != null && mPopWindow.isShowing()) {
mPopWindow.dismiss();
} else super.onBackPressed();
}
问题三:时间选择器弹出后挤压搜索窗 使其布局收缩的BUG。
如下直接改为设置window背景灰暗
Utility.setBackgroundAlpha(MyLogActivity.this, 1f);
/**
* 设置页面的透明度
*
* @param bgAlpha 1表示不透明
*/
public static void setBackgroundAlpha(Activity activity, float bgAlpha) {
WindowManager.LayoutParams lp = activity.getWindow().getAttributes();
lp.alpha = bgAlpha;
activity.getWindow().setAttributes(lp);
}
问题四:PopWindow的setTouchOutSide()方法无效的BUG。
解决同问题二
问题五: 由问题二,四所衍生出的Recyclerview 点击 长按事件 以及其他控件的点击事件均造成POP消失的bug。
解决如下 :在所有可能让POP消失的位置 加上
if (mPopWindow != null && mPopWindow.isShowing()) {
mPopWindow.dismiss();
return;
}
logRecyclerview.setOnScrollListener(new RecyclerView.OnScrollListener() {
@Override
public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
super.onScrollStateChanged(recyclerView, newState);
if (newState == AbsListView.OnScrollListener.SCROLL_STATE_TOUCH_SCROLL) {
if (mPopWindow != null && mPopWindow.isShowing()) {
mPopWindow.dismiss();
}
}
}
});