【问题标题】:Fixed view in ViewPager's content layout, under CoordinatorLayout修复了 ViewPager 内容布局中的视图,位于 CoordinatorLayout 下
【发布时间】:2016-05-27 22:36:56
【问题描述】:

我的应用有一个Activity 和一个TabLayout,还有3 个Fragments。在Activity's 布局中,我有CoordinatorLayoutViewPager。我还需要为 toolbar 制作动画。 现在在 Fragments 布局中,我需要在底部放置一个固定的 TextView。 下面给出的是活动和片段的XML

我面临的问题是在 Fragment 中修复了 TextView 布局在底部导航栏下方并且也在滚动, 我不想要的。

如何做到这一点?

activity.xml

<?xml version="1.0" encoding="utf-8"?>
<android.support.design.widget.CoordinatorLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/clMain"
    android:layout_width="match_parent"
    android:layout_height="match_parent">


    <android.support.design.widget.AppBarLayout
        android:id="@+id/appbar"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:theme="@style/AppTheme.AppBarOverlay">

        <android.support.v7.widget.Toolbar
            android:id="@+id/toolbar"
            android:layout_width="match_parent"
            android:layout_height="?attr/actionBarSize"
            android:background="?attr/colorPrimary"
            app:layout_scrollFlags="scroll|enterAlways|snap"
            app:popupTheme="@style/AppTheme.PopupOverlay">

            <TextView
                android:id="@+id/toolbar_title"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:text="@string/app_name"
                android:textColor="#FFFFFF"
                android:textSize="22sp"
                android:textStyle="bold" />
        </android.support.v7.widget.Toolbar>

        <android.support.design.widget.TabLayout
            android:id="@+id/tab_layout"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_below="@+id/toolbar"
            android:background="?attr/colorPrimary"
            android:minHeight="?attr/actionBarSize"
            android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar" />

    </android.support.design.widget.AppBarLayout>

    <android.support.v4.view.ViewPager
        android:id="@+id/pager"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        app:layout_behavior="@string/appbar_scrolling_view_behavior" />

</android.support.design.widget.CoordinatorLayout>

fragment.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout 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:background="@color/colorWhite"
    android:orientation="vertical">

    <android.support.v4.widget.NestedScrollView
        android:id="@+id/rlParent"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:layout_gravity="fill_vertical"
        android:layout_weight="1"
        android:fitsSystemWindows="true"
        android:orientation="vertical">


    </android.support.v4.widget.NestedScrollView>

    <TextView
        android:id="@+id/tvConfirmOrder"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:background="@color/colorDarkGreen"
        android:gravity="center"
        android:padding="10dp"
        android:text="@string/confirm_order"
        android:textColor="@color/colorWhite"
        android:textSize="18sp"
        android:textStyle="bold"></TextView>
</LinearLayout> 

【问题讨论】:

    标签: android android-viewpager material-design android-coordinatorlayout android-appbarlayout


    【解决方案1】:

    由于您希望每个片段的底部固定 TextView,因此可以通过将您的 TextView 转移到 activity.xml 并为其添加自定义行为来解决。

    所以,首先,创建一个代表自定义行为的类:

    public class FixedBottomViewBehavior extends CoordinatorLayout.Behavior<View> {
    
        public FixedBottomViewBehavior() {
        }
    
        public FixedBottomViewBehavior(Context context, AttributeSet attrs) {
            super(context, attrs);
        }
    
        @Override
        public boolean layoutDependsOn(CoordinatorLayout parent, View child, View dependency) {
            //making our bottom view depends on ViewPager (or whatever that have appbar_scrolling_view_behavior behavior)
            return dependency instanceof ViewPager;
        }
    
        @Override
        public boolean onDependentViewChanged(CoordinatorLayout parent, View child, View dependency) {
            if (ViewCompat.isLaidOut(parent)) {
                //attach our bottom view to the bottom of CoordinatorLayout
                child.setY(parent.getBottom() - child.getHeight());
    
                //set bottom padding to the dependency view to prevent bottom view from covering it
                dependency.setPadding(dependency.getPaddingLeft(), dependency.getPaddingTop(),
                        dependency.getPaddingRight(), child.getHeight());
            }
            return false;
        }
    } 
    

    这里没有魔法,我们只是让我们的底部视图依赖于带有appbar_scrolling_view_behavior(在我们的例子中是ViewPager)的一些视图,然后将它附加到CoordinatorLayout 的底部并设置一些填充依赖。

    其次,更改您的activity.xml,在此处添加您的TextView

    <?xml version="1.0" encoding="utf-8"?>
    <android.support.design.widget.CoordinatorLayout xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:app="http://schemas.android.com/apk/res-auto"
        android:id="@+id/clMain"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:fitsSystemWindows="true">
    
        <android.support.design.widget.AppBarLayout
            android:id="@+id/appbar"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:theme="@style/AppTheme.AppBarOverlay">
    
            <android.support.v7.widget.Toolbar
                android:id="@+id/toolbar"
                android:layout_width="match_parent"
                android:layout_height="?attr/actionBarSize"
                android:background="?attr/colorPrimary"
                app:layout_scrollFlags="scroll|enterAlways|snap"
                app:popupTheme="@style/AppTheme.PopupOverlay">
    
                <TextView
                    android:id="@+id/toolbar_title"
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content"
                    android:text="@string/app_name"
                    android:textColor="#FFFFFF"
                    android:textSize="22sp"
                    android:textStyle="bold" />
            </android.support.v7.widget.Toolbar>
    
            <android.support.design.widget.TabLayout
                android:id="@+id/tab_layout"
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                android:layout_below="@+id/toolbar"
                android:background="?attr/colorPrimary"
                android:minHeight="?attr/actionBarSize"
                android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar" />
    
        </android.support.design.widget.AppBarLayout>
    
        <android.support.v4.view.ViewPager
            android:id="@+id/pager"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            app:layout_behavior="@string/appbar_scrolling_view_behavior" />
    
        <TextView
            android:id="@+id/tvConfirmOrder"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_alignParentBottom="true"
            android:background="@color/colorDarkGreen"
            android:gravity="center"
            android:padding="10dp"
            android:text="@string/confirm_order"
            android:textColor="@color/colorWhite"
            android:textSize="18sp"
            android:textStyle="bold"
            app:layout_behavior="your.package.name.FixedBottomViewBehavior" />
    
    </android.support.design.widget.CoordinatorLayout>
    

    确保您已从 fragment.xml 中删除了您的 TextView。不要忘记将your.package.name.FixedBottomViewBehavior 更改为您的包名。

    瞧!这应该像一个具有适当工具栏动画和底部固定视图的魅力。

    另外:如果您不知道如何将您的 OnClick 事件传递给您的片段,您可以在您的 Activity 中按照这种方式进行操作: p>

        public interface OnConfirmOrderClickListener {
            void onConfirmOrderClick(View v);
        }
    
        public Fragment getActiveFragment() {
            String name = makeFragmentName(pager.getId(), pager.getCurrentItem());
            return getSupportFragmentManager().findFragmentByTag(name);
        }
    
        public String makeFragmentName(int viewId, int index) {
            return "android:switcher:" + viewId + ":" + index;
        } 
    
        tvConfirmOrder.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Fragment current = getActiveFragment();
                if (current instanceof OnConfirmOrderClickListener)
                    ((OnConfirmOrderClickListener) current).onConfirmOrderClick(v);
            }
        });
    

    别忘了制作你的片段implements OnConfirmOrderClickListener。希望对您有所帮助!

    【讨论】:

    • 如果固定视图只能在 viewpager 的一个片段中呢?
    • @VincentNOCK 根据屏幕上当前的片段隐藏/显示视图,使用onPageChangeListener 表示ViewPager
    • 我使用 onPageChangeListener 使按钮滚动。我将添加一个答案来解释如何做到这一点
    • 这个解决方案是最优雅的。谢谢@rom4ek
    • @rom4ek 如果我使用EditText 而不是TextView tvConfirmOrder,上述解决方案是否有效? (EditText 的高度根据输入的文字而变化。)
    【解决方案2】:

    对于那些希望按钮只出现在一个片段中的人,可以使用ViewPager 中的onPageChangeListener 来使按钮随着ViewPager 滚动

    viewPager.addOnPageChangeListener(new ViewPager.SimpleOnPageChangeListener() {
        @Override
        public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
            updateButtonPosition(position, positionOffsetPixels);
        }
    });
    
    
    //Move the button according to the tab you want it to be fixed. 
    //Here is the 3rd tab of a 3 tab viewPager, for instance
    private void updateButtonPosition(int position, int positionOffsetPixels) {
        int fixedRewardXPos;
        if (position == 0) {
            fixedRewardXPos = viewPager.getWidth() * 2 - positionOffsetPixels;
        } else if (position == 1) {
            fixedRewardXPos = viewPager.getWidth() - positionOffsetPixels;
        } else {
            fixedRewardXPos = -positionOffsetPixels;
        }
    
        //Add getLeft() to keep the button to its former position in it's fragment
        aButton.setX(fixedRewardXPos + aButton.getLeft());
    }
    

    【讨论】:

    • 您能否对不同位置值执行的操作添加一点解释?
    猜你喜欢
    • 2015-08-28
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2017-02-02
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多