当我在另一个片段(我们称之为主片段)上显示一个片段(它是具有 #77000000
背景的全屏)时,我的主片段仍然对点击做出反应(即使我们没有看到它也可以点击一个按钮) .
问题:如何防止点击第一个(主)片段?
编辑
不幸的是,我不能只隐藏主要片段,因为我在第二个片段上使用透明背景(因此,用户可以看到位于后面的内容)。
main
Fragment
的 Visibility
设置为 GONE
。
onClick
方法不返回任何内容。但是你给出一个想法,谢谢(我会尽快发布答案)。
将第二个片段视图的 clickable
属性设置为 true。视图将捕获事件,以便它不会传递给主片段。所以如果第二个片段的视图是一个布局,这将是代码:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:clickable="true" />
解决方案非常简单。在我们的第二个片段(与我们的主片段重叠)中,我们只需要捕获 onTouch
事件:
@Override
public View onCreateView(LayoutInflater inflater,ViewGroup container,Bundle savedInstance){
View root = somehowCreateView();
/*here is an implementation*/
root.setOnTouchListener(new View.OnTouchListener() {
public boolean onTouch(View v, MotionEvent event) {
return true;
}
});
return root;
}
只需将 clickable="true"
和 focusable="true"
添加到父布局
<android.support.constraint.ConstraintLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:clickable="true"
android:focusable="true">
<!--Your views-->
</android.support.constraint.ConstraintLayout>
如果您使用 AndroidX
,试试这个
<androidx.constraintlayout.widget.ConstraintLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:clickable="true"
android:focusable="true">
<!--Your views-->
</androidx.constraintlayout.widget.ConstraintLayout>
focuseable
并不是必需的。
focusable="true"
只是为了避免在 Android Studio 中出现警告。
如果两个片段放置在同一个容器视图中,则在显示第二个片段时应该隐藏第一个片段。
如果想了解更多关于如何解决 Fragment 问题的问题,可以看我的库:https://github.com/JustKiddingBaby/FragmentRigger
FirstFragment firstfragment;
SecondFragment secondFragment;
FragmentManager fm;
FragmentTransaction ft=fm.beginTransaction();
ft.hide(firstfragment);
ft.show(secondFragment);
ft.commit();
您需要将 android:focusable="true"
与 android:clickable="true"
添加
Clickable
表示它可以被指针设备点击或被触摸设备点按。
Focusable
表示它可以从键盘等输入设备获得焦点。像键盘这样的输入设备无法根据输入本身决定将其输入事件发送到哪个视图,因此它们将它们发送到具有焦点的视图。
方法一:
您可以添加到所有片段布局
android:clickable="true"
android:focusable="true"
android:background="@color/windowBackground"
方法 2:(以编程方式)
从 FragmentBase
等扩展所有片段。然后将此代码添加到 FragmentBase
@Override
public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) {
super.onViewCreated(view, savedInstanceState);
getView().setBackgroundColor(getResources().getColor(R.color.windowBackground));
getView().setClickable(true);
getView().setFocusable(true);
}
我们中的一些人为这个线程贡献了不止一个解决方案,但我还想提一个其他解决方案。如果您不喜欢像我一样在 XML 中将可点击和可聚焦等于 true 的每个布局的根 ViewGroup。如果你有一个像下面这样的,你也可以把它放在你的基地;
override fun onCreateView(
inflater: LayoutInflater,
container: ViewGroup?,
savedInstanceState: Bundle?
) : View? {
super.onCreateView(inflater, container, savedInstanceState)
val rootView = inflater.inflate(layout, container, false).apply {
isClickable = true
isFocusable = true
}
return rootView
}
您也可以使用内联变量,但出于个人原因我不喜欢它。
我希望它对那些讨厌布局 XML 的人有所帮助。
可接受的答案将“起作用”,但也会导致性能成本(过度绘制,重新测量方向变化),因为底部的片段仍在绘制中。也许您应该简单地通过标签或 ID 找到片段,并在需要再次显示时将可见性设置为 GONE 或 VISIBLE。
在科特林:
fragmentManager.findFragmentByTag(BottomFragment.TAG).view.visibility = GONE
当您使用动画时,此解决方案优于 FragmentTransaction
的替代 hide()
和 show()
方法。您只需从 Transition.TransitionListener
的 onTransitionStart()
和 onTransitionEnd()
调用它。
您可以做的是,您可以通过使用 onClick 属性为该主片段的父布局提供空白点击到前一个片段的布局,并且在活动中您可以创建一个函数 doNothing(View view)
并且不要在其中写入任何内容。这将为您完成。
这听起来像是 DialogFragment 的一个案例。否则,使用 Fragment Manager 提交一个隐藏,另一个显示。这对我有用。
android:clickable="true"
的添加对我不起作用。当 CoordinatorLayout 是父布局时,此解决方案不适用于它。这就是为什么我将RelativeLayout 作为父布局,添加android:clickable="true"
并将CoordinatorLayout 放置在这个RelativeLayout 上。
我有多个具有相同 xml 的片段。
花了几个小时后,我删除了 setPageTransformer
并开始工作
// viewpager.setPageTransformer(false, new BackgPageTransformer())
我有严格的逻辑。
public class BackgPageTransformer extends BaseTransformer {
private static final float MIN_SCALE = 0.75f;
@Override
protected void onTransform(View view, float position) {
//view.setScaleX Y
}
@Override
protected boolean isPagingEnabled() {
return true;
}
}
RelativeLayout
,并使用clickeable
属性设置了整个视图。 @Dmitry 的解决方案解决了我的问题。