У меня есть 2 ScrollViews в моем макете android. Как синхронизировать их положения прокрутки?
Синхронизация позиций прокрутки ScrollView - android
Ответ 1
В ScrollView есть метод...
protected void onScrollChanged(int x, int y, int oldx, int oldy)
К сожалению, Google никогда не думал, что нам нужно будет получить к нему доступ, поэтому они сделали его защищенным и не добавили крюк setOnScrollChangedListener. Поэтому мы должны будем сделать это для себя.
Сначала нам нужен интерфейс.
package com.test;
public interface ScrollViewListener {
void onScrollChanged(ObservableScrollView scrollView, int x, int y, int oldx, int oldy);
}
Затем нам нужно переопределить класс ScrollView, чтобы обеспечить крючок ScrollViewListener.
package com.test;
import android.content.Context;
import android.util.AttributeSet;
import android.widget.ScrollView;
public class ObservableScrollView extends ScrollView {
private ScrollViewListener scrollViewListener = null;
public ObservableScrollView(Context context) {
super(context);
}
public ObservableScrollView(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
}
public ObservableScrollView(Context context, AttributeSet attrs) {
super(context, attrs);
}
public void setScrollViewListener(ScrollViewListener scrollViewListener) {
this.scrollViewListener = scrollViewListener;
}
@Override
protected void onScrollChanged(int x, int y, int oldx, int oldy) {
super.onScrollChanged(x, y, oldx, oldy);
if(scrollViewListener != null) {
scrollViewListener.onScrollChanged(this, x, y, oldx, oldy);
}
}
}
И мы должны указать этот новый класс ObservableScrollView в макете вместо существующих тегов ScrollView.
<com.test.ObservableScrollView
android:id="@+id/scrollview1"
... >
...
</com.test.ObservableScrollView>
Наконец, мы все вместе ставим в классе Layout.
package com.test;
import android.app.Activity;
import android.os.Bundle;
public class Q3948934 extends Activity implements ScrollViewListener {
private ObservableScrollView scrollView1 = null;
private ObservableScrollView scrollView2 = null;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.q3948934);
scrollView1 = (ObservableScrollView) findViewById(R.id.scrollview1);
scrollView1.setScrollViewListener(this);
scrollView2 = (ObservableScrollView) findViewById(R.id.scrollview2);
scrollView2.setScrollViewListener(this);
}
public void onScrollChanged(ObservableScrollView scrollView, int x, int y, int oldx, int oldy) {
if(scrollView == scrollView1) {
scrollView2.scrollTo(x, y);
} else if(scrollView == scrollView2) {
scrollView1.scrollTo(x, y);
}
}
}
Код scrollTo() заботится о любых условиях цикла для нас, поэтому нам не нужно об этом беспокоиться. Единственное предостережение в том, что это решение не гарантируется для работы в будущих версиях Android, потому что мы переопределяем защищенный метод.
Ответ 2
Улучшение решения Andy: В его коде он использует scrollTo, проблема в том, что если вы бросаете один прокрутки в одном направлении, а затем бросаете еще один в другом направлении, вы заметите, что первый не останавливает его предыдущее движение движения.
Это связано с тем, что scrollView использует computeScroll() для выполнения своих жестов, и он вступает в конфликт с scrollTo.
Чтобы предотвратить это, просто запрограммируйте onScrollChanged следующим образом:
public void onScrollChanged(ObservableScrollView scrollView, int x, int y, int oldx, int oldy) {
if(interceptScroll){
interceptScroll=false;
if(scrollView == scrollView1) {
scrollView2.onOverScrolled(x,y,true,true);
} else if(scrollView == scrollView2) {
scrollView1.onOverScrolled(x,y,true,true);
}
interceptScroll=true;
}
}
с interceptScroll статический булев, инициализированный в true. (это помогает избежать бесконечных циклов на ScrollChanged)
onOverScrolled - единственная функция, которую я нашел, которая могла бы использоваться, чтобы остановить прокрутку scrollView (но могут быть и другие, которых я пропустил!)
Чтобы получить доступ к этой функции (которая защищена), вы должны добавить ее в свой ObservableScrollViewer
public void onOverScrolled(int scrollX, int scrollY, boolean clampedX, boolean clampedY) {
super.onOverScrolled(scrollX, scrollY, clampedX, clampedY);
}
Ответ 3
Почему бы просто не реализовать OnTouchListener
в своей деятельности.
Затем переопределите метод onTouch, затем получите начальное положение прокрутки первого ScrollViewOne.getScrollY()
и обновите ScrollViewTwo.scrollTo(0, ScrollViewOne.getScrollY());
Еще одна идея...:)
Ответ 4
В пакете Android support-v4 Android предоставляет новый класс с именем NestedScrollView
.
мы можем заменить <ScrollView>
node на <android.support.v4.widget.NestedScrollView>
в макете xml,
и реализует свой NestedScrollView.OnScrollChangeListener
в Java для обработки прокрутки.
Это облегчает процесс.