在 Android 中创建锁定的 ScrollView

Posted

技术标签:

【中文标题】在 Android 中创建锁定的 ScrollView【英文标题】:Creating a locked ScrollView in Android 【发布时间】:2017-12-13 22:26:34 【问题描述】:

我正在尝试在 android 中实现ScrollView,当在当前滚动位置上方添加项目时不会滚动。

ScrollView 的默认实现如下:

在当前滚动位置上方添加一个项目:

在当前滚动位置下方添加一个项目:

如何在当前滚动位置上方添加项目之前“锁定”ScrollView

这是我的布局文件,我目前已经覆盖了ScrollViewLinearLayout,但还没有进行任何更改。

<LinearLayout
      android:layout_
      android:orientation="vertical"
      android:layout_>

    <LinearLayout
        android:id="@+id/LinearLayout02"
        android:layout_
        android:layout_
        android:layout_alignParentBottom="true">
        <Button
            android:id="@+id/Button02"
            android:layout_
            android:layout_
            android:layout_weight="1" android:text="Add To Top"
            android:onClick="addToStart">
        </Button>
        <Button
            android:id="@+id/Button03"
            android:layout_
            android:layout_
            android:layout_weight="1"
            android:text="Add to End"
            android:onClick="addToEnd">
        </Button>
    </LinearLayout>
    <com.poc.scroller.locable.lockablescrollerpoc.LockedScrollView xmlns:android="http://schemas.android.com/apk/res/android"
        android:layout_
        android:layout_
        android:fillViewport="true"
        android:scrollbarAlwaysDrawVerticalTrack="true"
        android:verticalScrollbarPosition="right"
        android:fadeScrollbars="false"
        android:background="@color/scrollColor">

        <com.poc.scroller.locable.lockablescrollerpoc.LockedLinearLayout
            android:layout_
            android:layout_
            android:orientation="vertical"
            android:id="@+id/Container">
        </com.poc.scroller.locable.lockablescrollerpoc.LockedLinearLayout>

    </com.poc.scroller.locable.lockablescrollerpoc.LockedScrollView>
</LinearLayout>
</android.support.constraint.ConstraintLayout>

示例源代码: https://github.com/Amaros90/android-lockable-scroller-poc

谢谢!

【问题讨论】:

请提供布局sn-p 尝试在按钮点击时调用滚动到顶部并滚动到底部 @AjayVenugopal 你的建议是给应用程序片状行为 @surajshinde 完成! 是否有特殊原因,您正在使用 ScrollView?使用RecyclerView 是我的方法,因为它用于解决您的问题。 【参考方案1】:

您可以通过使用LinearLayoutaddOnLayoutChangeListener 并重置ScrollViewScrollY 轻松获得相反的行为。这是您的ScrollViewActivity.onCreate

中的实现
protected void onCreate(Bundle savedInstanceState) 
    super.onCreate(savedInstanceState);
    setContentView(R.layout.scrollview);

    _linearLayout = (LockedLinearLayout)findViewById(R.id.Container);
    _scrollView = (LockedScrollView)findViewById(R.id.ScrollView);
    _layoutInflater = (LayoutInflater)getSystemService(Context.LAYOUT_INFLATER_SERVICE);

    _linearLayout.addOnLayoutChangeListener(new View.OnLayoutChangeListener() 
        @Override
        public void onLayoutChange(View v, int left, int top, int right, int bottom, int oldLeft, int oldTop, int oldRight, int oldBottom) 
            _scrollView.scrollTo(0, _scrollView.getScrollY() + (bottom - oldBottom ));
        
    );

    addExampleImage(10, _linearLayout);

然后您可以轻松地标记addToEnd 函数或检查添加子项的位置以避免在将某些子项添加到底部时更改滚动。

【讨论】:

【参考方案2】:

您可以尝试使用RecyclerView 并实现onDataSetChanged()。然后检测是否按下了add to TOPadd to END 按钮。然后使用scrollToPositionWithOffset(int, int) 管理滚动。

例如:

//Scroll item 3 to 20 pixels from the top
linearLayoutManager.scrollToPositionWithOffset(3, 20);

为了恢复RecyclerView的滚动位置,这是保存滚动位置的方法(scrollToPositionWithOffset(int, int)方法的两个参数):

int index = linearLayoutManager.findFirstVisibleItemPosition(); 
View v = linearLayoutManager.getChildAt(0); 
int top = (v == null) ? 0 : (v.getTop() - linearLayoutManager.getPaddingTop())

【讨论】:

使用RecyclerView时可以调用该函数,滚动不会改变_flagsAdapter.notifyItemInserted(0);但是我们仍然需要用ScrollView而不是RecyclerView来实现这个功能【参考方案3】:

实施这对我有用。您可以查看我在原始问题中添加的示例应用程序。

public class LockableScrollView extends ScrollView 

    private boolean _enabled = true;

    public LockableScrollView(Context context, AttributeSet attrs) 
        super(context, attrs);
        super.setFillViewport(true);
    

    @Override
    public void addView(View layout, ViewGroup.LayoutParams params) 
        super.addView(layout, params);

        ((ViewGroup)layout).setOnHierarchyChangeListener(new OnHierarchyChangeListener() 
            @Override
            public void onChildViewAdded(View layout, View item) 
                if (_enabled) 
                    item.addOnLayoutChangeListener(new View.OnLayoutChangeListener() 
                        @Override
                        public void onLayoutChange(View item, int left, int top, int right, int bottom, int oldLeft, int oldTop, int oldRight, int oldBottom) 
                        item.removeOnLayoutChangeListener(this);

                        int scrollViewY = ((LockableScrollView)item.getParent().getParent()).getScrollY();
                        int layoutPosition = ((View)item.getParent()).getTop();
                        boolean shouldScroll = item.getTop() + layoutPosition <= scrollViewY || item.getBottom() + getTop() <= scrollViewY;



                        if (shouldScroll) 
                            final int childViewHeight = item.getHeight();

                            ((View)item.getParent()).addOnLayoutChangeListener(new View.OnLayoutChangeListener() 
                                @Override
                                public void onLayoutChange(View layout, int left, int top, int right, int bottom, int oldLeft, int oldTop, int oldRight, int oldBottom) 
                                    layout.removeOnLayoutChangeListener(this);

                                    LockableScrollView scrollView = ((LockableScrollView)layout.getParent());
                                    scrollView.scrollTo(scrollView.getScrollX(), scrollView.getScrollY() + childViewHeight);
                                
                            );
                        
                        
                    );
                
            

            @Override
            public void onChildViewRemoved(View layout, View item) 
                if (_enabled) 
                    int scrollViewY = ((LockableScrollView)layout.getParent()).getScrollY();
                    int layoutPosition = layout.getTop();
                    boolean shouldScroll = item.getTop() + layoutPosition <= scrollViewY || item.getBottom() + getTop() <= scrollViewY;


                    if (shouldScroll) 
                        final int childViewHeight = item.getHeight();

                        layout.addOnLayoutChangeListener(new View.OnLayoutChangeListener() 
                            @Override
                            public void onLayoutChange(View layout, int left, int top, int right, int bottom, int oldLeft, int oldTop, int oldRight, int oldBottom) 
                                layout.removeOnLayoutChangeListener(this);

                                LockableScrollView scrollView = ((LockableScrollView)layout.getParent());
                                scrollView.scrollTo(scrollView.getScrollX(), scrollView.getScrollY() - childViewHeight);
                            
                        );
                    
                
            
        );
    

【讨论】:

【参考方案4】:

只要得到它就可以轻松做到..

首先在 onclick 事件中存储 int 值 使用 getFirstVisiblePosition()

例子。

点击添加元素按钮时这样做---

int currentpos = recycleview.getFirstVisiblePosition();

现在在您将元素插入列表/recyclerview 之后---

recyclerview.scrolltoposition(currentpos);

如果你想做到没有错误,那就试试吧..

祝你好运:)

【讨论】:

以上是关于在 Android 中创建锁定的 ScrollView的主要内容,如果未能解决你的问题,请参考以下文章

如何在 EXTJS 4.1 中创建导航栏?

在一定的空闲时间后锁定 android 应用程序

从NPM添加bootrap反应在Visual Studio中创建另一个程序包锁定文件

如何使用 htaccess 文件锁定文件夹?

只读SAS视图(双击也是如此)

如何在android中创建链接按钮