将片段手势与ViewPager一起应用在片段级别,但其默认片段已禁用

问题描述 投票:7回答:1

我发现了以下问题:Android: FragmentActivity in FragmentActivity (Scrollview in Navigationbar)。但是,我的问题是关于如何使用事务显示片段以及以安全方式启用滑动手势检测的问题。注意:我发布的答案中有一个链接(这也具有使用事务显示容器中片段的有效方法,包括两种情况)。请看。我已经尝试过一些方面,但是通过使用支持ViewPager而不是嵌套的片段:

详细信息:https://moqups.com/[email protected]/lc0ZOERO/p:a5d7b55eb

  1. 使用自定义ViewPager使用onTouchEvent和onInterceptTouchEvent禁用默认滑动。
  2. 声明的FragmentStatePagerAdapter带有片段列表,每个片段显示在每个选项卡中。
  3. 在main_activity的main_layout中声明片段,该片段与ViewPager并排放置。想法是在单击ViewPager_fragment上的按钮并在用户按下return时显示activity_fragments按钮,然后使用添加时的向后堆栈事务并在BackPressed上将其弹出时,再次显示默认的ViewPager_fragment活动。因此,我还将维护自己的自定义后堆栈,以在后堆栈显示/隐藏activity_fragments弹出交易。

现在,我要实现的是仅对右[从左到右]滑动使用滑动手势来完成以上第三点。

我为此使用了GestureListener和SimpleOnGestureListener以及活动的OnTouchEvent。

我面临的问题是:

此手势在活动屏幕上位于片段和活动部分下方的部分上起作用。我想要手势工作

  1. 在其布局中具有多个视图的片段区域上。
  2. 仅在左右方向上。
  3. 仅从activity_fragment到ViewPager的选项卡片段,其行为就像历史导航一样,已在onBakPressed / onKeyDown中完成使用后堆栈弹出窗口和我的自定义后堆栈实现。

我尝试了以下课程,并在这样的布局中更改了活动片段。

我的手势和触摸侦听器扩展了框架布局:

        public class OnSwipeTouchListener extends FrameLayout {

        private final GestureDetector gestureDetector;
        private static final String TAG = "OnSwipeTouchListener";
    private Context context;
        public OnSwipeTouchListener(Context context, AttributeSet attrs) {
            super(context, attrs);
            this.context=context;
            gestureDetector = new GestureDetector(context, new GestureListener());
            setClickable(true);

        }

        @Override
        public boolean onInterceptTouchEvent(MotionEvent ev) {
            return super.onInterceptTouchEvent(ev);
        }

        @Override
        public boolean onTouchEvent(MotionEvent event) {
 gestureDetector.onTouchEvent(motionEvent);
            return super.onTouchEvent(event);
        }


        private final class GestureListener extends SimpleOnGestureListener {

            private static final int SWIPE_THRESHOLD = 100;
            private static final int SWIPE_VELOCITY_THRESHOLD = 100;

            @Override
            public boolean onDown(MotionEvent e) {

                return true;
            }

            @Override
            public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) {

                boolean result = false;
                try {
                    float diffY = e2.getY() - e1.getY();
                    float diffX = e2.getX() - e1.getX();
                    if (Math.abs(diffX) > Math.abs(diffY)) {
                        if (Math.abs(diffX) > SWIPE_THRESHOLD && Math.abs(velocityX) > SWIPE_VELOCITY_THRESHOLD) {
                            if (diffX > 0) {
                                onSwipeRight();
                            } else {
                                onSwipeLeft();
                            }
                        }
                    } else {
                        if (Math.abs(diffY) > SWIPE_THRESHOLD && Math.abs(velocityY) >           SWIPE_VELOCITY_THRESHOLD) {
                            if (diffY > 0) {
                                onSwipeBottom();
                            } else {
                                onSwipeTop();
                            }
                        }
                    }
                } catch (Exception exception) {
                    //see that e1 is null
                }
                return result;
            }
        }

        public void onSwipeRight() {
            ///manage my back stack history here.
        }

        public void onSwipeLeft() {
        }

        public void onSwipeTop() {
        }

        public void onSwipeBottom() {
        }
    }

然后将框架布局更改为此类参考:

<path.to.my.package.OnSwipeTouchListener
            android:id="@+id/activity_frag_frame_layout1"
            android:layout_width="fill_parent"
            android:layout_height="fill_parent"
            android:visibility="invisible" >

            <fragment
                android:id="@+id/activity_frag1"
                android:layout_width="fill_parent"
                android:layout_height="fill_parent"
                class="path.to.my.package.fragments.Activity_Frag1"
                android:tag="act_frag1" />
        </path.to.my.package.OnSwipeTouchListener>

它对我没有多大帮助,所以应该对您有所帮助。在其中一个片段中有一个searchView停止工作,并且没有单击起作用。如果您能达到我想要的目标,请在这方面提供帮助。

更新1:在onTouchEvent和onInterceptTouchEvent中都返回true可以达到预期的效果,但是它会阻止SearchView所在的片段中的单击,在可单击的视图中没有单击起作用。仅滑动有效。更新:在我的最新文章中:我也完全放弃了使用事务后退堆栈,因为我依赖于我自定义的后退堆栈,其中我维护所有选项卡的导航历史记录。我使用ViewPager和FragmentStatePagerAdapter给出的setCurrentTab和onTabSelected方法。

更新2:这是很难实现的:检查哪些事件需要拦截,哪些事件将传递给子视图:http://developer.android.com/training/gestures/viewgroup.html

更新3:

  1. 在我的活动布局中,有一个视图寻呼机,在它下面的每个框架布局中有一个片段。

  2. [启动应用程序时,ViewPager在第一个选项卡中显示该片段。

  3. 单击此片段上的按钮时,将显示视图分页器下面的帧布局中的一个片段,并将其添加到我的自定义反向堆栈中。

  4. 当按下返回键时,此片段再次被隐藏以在视图寻呼机中显示选项卡片段。以上所有步骤均已完成。只有我想要左右滑动手势才能工作,我发现可以通过拦截触摸事件来做到这一点。但这根本不做。当我返回true时,只有滑动手势有效,否则此操作将被取消,点击此片段即可。

android android-fragments android-viewpager swipe-gesture
1个回答
1
投票

我在http://mobi-app-dev.blogspot.in/2014/01/android-transactions.html中提到了这个答案。

检查哪些事件需要拦截,哪些事件将传递给子视图:http://developer.android.com/training/gestures/viewgroup.html

而不是在屏幕上的框架布局中使用我的滑动手势监听器附有哪些片段,现在我在相对布局上使用它,这是片段。

@Override
    public boolean onInterceptTouchEvent(MotionEvent event) {
        boolean intercepted = super.onInterceptTouchEvent(event);

        // In general, we don't want to intercept touch events. They should be
        // handled by the child view.

        gestureDetector.onTouchEvent(event);
        return false;
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        boolean touched = super.onTouchEvent(event);
        gestureDetector.onTouchEvent(event);
        return touched;
    }

以下是我在活动中使用片段时要遵循的一些技巧:

有效管理碎片交易:

[注意:如果您使用的是FragmentStatePagerAdapter,则可以使用在该适配器中用作数据的片段列表中特定索引处的另一个索引替换另一个片段的方式,相同的标签,但其内容被更改而未更改标签。然后使用notifyDataSetChanged()。在这里,您不使用事务,但是您可以使用]。

  1. 使用Fragment.instantiate()初始化您的片段,而不是构造函数。
  2. 使用侦听器更新/删除实例的引用(来自用于引用它们的集合)片段onCreateView和onDestroy。
  3. 使用getView()获取片段的根视图,无论何时何地以及更新功能。
  4. 不要跟踪实例,实际上应尽可能避免使用静态指针。
  5. 使用getSupportFragment而不是保留FragmentManager的引用在视图寻呼机适配器或其他位置。
  6. 在事务隐藏视图上使用附加分隔符,以消除片段的根视图并重新初始化。这是因为一次只需要一个片段可见,而另一个则需要分离。因此,按需重新连接。
  7. getActivity()或getApplicationContext()(如果适用),而不是保留一个全局变量以保留对上下文的引用。
  8. onConfiguration更改(方向,键盘,调整大小):在活动中进行并传递到管理它的活动片段。重要说明:如果确实要使用片段进行事务处理,请不要在布局中声明它们。在添加片段事务中提供容器(布局)ID。
  9. 片段事务应在事件调用中完成,而不是在其他任何地方进行,并且此类事件不应重复。这样可以防止非法参数异常。
  10. 有一个ViewPager方法,您不必使用Fragments。相反,您可以使用“视图”或“视图组”。这将替换片段的嵌套。
  11. 检测片段的级别,这意味着如果您想用片段'B'替换现有的片段'A',并且当您想在按回'B'时返回以显示片段'A'时,请进行该交易在后面的堆栈中。

提示:添加滑动功能时要小心,因为还有其他可单击的视图。在RootView上使用拦截触摸轻扫,返回假但分析触摸。


另外一件事:您应该观察到一次在一个容器中有一个片段可以解决另一个问题:如果有两个片段,一个在另一个片段的顶部,则来自一个片段的点击会转到下面的另一个片段。因此,隐藏或替换一次只能有一个片段。 Back-Stack帮助维护导航级别,并在更改或更新配置时,尝试更新目标活动片段的相同实例,而不是执行事务,因为这会改变所导航片段的顺序。

© www.soinside.com 2019 - 2024. All rights reserved.