Прокрутите до последней строки TableLayout в ScrollView - PullRequest
12 голосов
/ 21 июня 2010

Я хочу иметь динамическую таблицу со строками, добавляемыми с течением времени в результате взаимодействия с пользователем, используя TableLayout внутри ScrollView.Это прекрасно работает, но когда я хочу прокрутить до конца таблицы, используя fullScroll(), она всегда пропускает последнюю строку;то есть, он прокручивается так, что виден последний перед последним.Последняя строка видна при ручной прокрутке, и полоса прокрутки тоже правильная.

Я, конечно, открыт для предложений о том, как сделать из этого лучший макет;но мне особенно интересно понять, почему fullScroll() ведет себя так.Должен ли я дать ему другой параметр или использовать что-то совсем другое?Или это происходит потому, что добавленная строка еще не видна?(если так, как я могу решить это?) Или я пропустил какую-то другую очевидную вещь?

Следующий код повторяет проблему:

TestActivity.java:

package com.example.android.tests;

import java.util.Random;
import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.ScrollView;
import android.widget.TableLayout;
import android.widget.TableRow;
import android.widget.TextView;

public class TestActivity extends Activity {
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        ((Button) findViewById(R.id.AddRow)).setOnClickListener(new View.OnClickListener() {
            public void onClick(View v) {
                Random rnd = new Random();
                TableRow nr = new TableRow(v.getContext());
                for (int c=0; c<3; c++) {
                    TextView nv = new TextView(v.getContext());
                    nv.setText(Integer.toString(rnd.nextInt(20)-10)); 
                    nr.addView(nv);
                }
                ((TableLayout) findViewById(R.id.Table)).addView(nr);
                // Scrolls to line before last - why?
                ((ScrollView) findViewById(R.id.TableScroller)).fullScroll(View.FOCUS_DOWN);
            }
        });

    }
}

main.xml:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
  xmlns:android="http://schemas.android.com/apk/res/android"
  android:orientation="vertical"
  android:layout_width="fill_parent"
  android:layout_height="fill_parent">
  <Button
    android:text="Add Row"
    android:id="@+id/AddRow"
    android:layout_width="fill_parent"
    android:layout_height="wrap_content"
    android:layout_alignParentBottom="true" />
  <ScrollView
    android:id="@+id/TableScroller"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:layout_above="@id/AddRow"
    android:layout_alignParentTop="true" >
    <TableLayout
      android:id="@+id/Table"
      android:layout_width="fill_parent"
      android:layout_height="wrap_content"
      android:stretchColumns="0,1,2" />
  </ScrollView>
</RelativeLayout>

Редактировать: для справки, я реализовал решение Romain Guy следующим образом:

В TestActivity.java,замените:

            // Scrolls to line before last - why?
            ((ScrollView) findViewById(R.id.TableScroller)).fullScroll(View.FOCUS_DOWN);

на:

            // Enqueue the scrolling to happen after the new row has been layout
            ((ScrollView) findViewById(R.id.TableScroller)).post(new Runnable() {
                public void run() {
                    ((ScrollView) findViewById(R.id.TableScroller)).fullScroll(View.FOCUS_DOWN);
                }

            });

Что отлично работает.

Ответы [ 2 ]

15 голосов
/ 22 июня 2010

В то время, когда вы выполняете свой fullScroll (), макет еще не произошел, поэтому ScrollView использует «старый» размер таблицы.Вместо того, чтобы сразу вызывать fullScroll (), используйте View.post (Runnable).

2 голосов
/ 22 марта 2012

Считая подсказку выше полезной, вот простая реализация, которая прокручивает ScrollView, чтобы сделать видимого дочернего элемента видимым ...

a: Подготовьте следующий вспомогательный класс

public class ScrollToTrick implements Runnable {

ScrollView scroller;
View       child;

ScrollToTrick(ScrollView scroller, View child) {
    this.scroller=scroller; 
    this.child=child;

}
public void run() {
    scroller.scrollTo(0, child.getTop());
}
}

б) назови это так

my_scroller.post(new ScrollToTrick(my_scroller,child_to_scroll_to) );
...