ChatGPT解决这个技术问题 Extra ChatGPT

另一个片段问题的片段

当我在另一个片段(我们称之为主片段)上显示一个片段(它是具有 #77000000 背景的全屏)时,我的主片段仍然对点击做出反应(即使我们没有看到它也可以点击一个按钮) .

问题:如何防止点击第一个(主)片段?

编辑

不幸的是,我不能只隐藏主要片段,因为我在第二个片段上使用透明背景(因此,用户可以看到位于后面的内容)。

根据您提供给我们使用的内容,您应该在不使用时尝试将 main FragmentVisibility 设置为 GONE
在没有看到您如何实现 onClicked 方法的情况下,我猜您在单击时返回“false”。
@DeeV,onClick 方法不返回任何内容。但是你给出一个想法,谢谢(我会尽快发布答案)。
哦。你是对的。 onTouch 返回它。我只是希望我能理解为什么一个触摸事件会从一个片段中消失。如果您不发出触摸事件,则不应这样做。
@DeeV,看起来如果您的视图(例如在其他视图之上)没有捕获 onTouch 事件,那么系统会继续搜索具有相同坐标的其他视图。

J
Jure Vizjak

将第二个片段视图的 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" />

这对我有用。这似乎比@Dmitry Zaitsev 上面给出的解决方案更容易。有什么理由说明这是一个坏主意吗?我似乎什么都想不出来,但我只是想确定一下。
这对我不起作用。我在片段中有一个 RelativeLayout,并使用 clickeable 属性设置了整个视图。 @Dmitry 的解决方案解决了我的问题。
这对我有用。 Android 中的“可点击”显然有点像 iOS 的“userInteractionEnabled”
为什么 Android 会让我们承受苛刻的编码条件?!
我对 ViewPager 有同样的问题。当我在第一页滚动时,它也传递到第二页,这个解决方案对我不起作用。有任何想法吗?
D
Dmitry Zaytsev

解决方案非常简单。在我们的第二个片段(与我们的主片段重叠)中,我们只需要捕获 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;
}

你的解决方案为此工作+1,但你能告诉我为什么我们需要明确地这样做吗?
@Hunt 你不必这样做。这只是另一种方法(见接受的答案)
android:clickable="true" 在您的第二个片段的 xml 主布局上。
这个解决方案有一个问题,当您进入可访问性对讲模式时,它不会读取单个元素,而是将焦点转移到根视图。
Kotlin 版本:root.setOnTouchListener { _, _ -> true}
M
Miravzal

只需将 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 中出现警告。
J
JingYeoh

如果两个片段放置在同一个容器视图中,则在显示第二个片段时应该隐藏第一个片段。

如果想了解更多关于如何解决 Fragment 问题的问题,可以看我的库:https://github.com/JustKiddingBaby/FragmentRigger

FirstFragment firstfragment;
SecondFragment secondFragment;
FragmentManager fm;
FragmentTransaction ft=fm.beginTransaction();
ft.hide(firstfragment);
ft.show(secondFragment);
ft.commit();

这应该是正确的答案!谢谢你,兄弟。我已经在另一个项目中解决了这个问题。但我忘记了我是如何解决的,最后,我得到了你的答案。谢谢你。
我认为这不是正确的解决方案。片段/活动在视图堆栈中工作。将顶部片段弹出堆栈后,您必须再次调用 .show ,这意味着必须通知底部片段顶部的片段已消失。它只是添加了额外的逻辑来维护。
H
Hossam Hassan

您需要将 android:focusable="true"android:clickable="true" 添加

Clickable 表示它可以被指针设备点击或被触摸设备点按。

Focusable 表示它可以从键盘等输入设备获得焦点。像键盘这样的输入设备无法根据输入本身决定将其输入事件发送到哪个视图,因此它们将它们发送到具有焦点的视图。


并且需要 focusable="true" 以避免在新的 Android Studio 中出现警告
M
MarsPeople

方法一:

您可以添加到所有片段布局

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);
}

Y
Yekta Sarıoğlu

我们中的一些人为这个线程贡献了不止一个解决方案,但我还想提一个其他解决方案。如果您不喜欢像我一样在 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 的人有所帮助。


A
Allan Veloso

可接受的答案将“起作用”,但也会导致性能成本(过度绘制,重新测量方向变化),因为底部的片段仍在绘制中。也许您应该简单地通过标签或 ID 找到片段,并在需要再次显示时将可见性设置为 GONE 或 VISIBLE。

在科特林:

fragmentManager.findFragmentByTag(BottomFragment.TAG).view.visibility = GONE

当您使用动画时,此解决方案优于 FragmentTransaction 的替代 hide()show() 方法。您只需从 Transition.TransitionListeneronTransitionStart()onTransitionEnd() 调用它。


M
Maximilian Ast

您可以做的是,您可以通过使用 onClick 属性为该主片段的父布局提供空白点击到前一个片段的布局,并且在活动中您可以创建一个函数 doNothing(View view) 并且不要在其中写入任何内容。这将为您完成。


J
Juan Mendez

这听起来像是 DialogFragment 的一个案例。否则,使用 Fragment Manager 提交一个隐藏,另一个显示。这对我有用。


Z
Zhebzhik Babich

android:clickable="true" 的添加对我不起作用。当 CoordinatorLayout 是父布局时,此解决方案不适用于它。这就是为什么我将RelativeLayout 作为父布局,添加android:clickable="true" 并将CoordinatorLayout 放置在这个RelativeLayout 上。


A
AskQ

我有多个具有相同 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;
    }
}