質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

87.37%

Android ScrollViewでスクロールが止まった時に処理を行いたい

解決済

回答 2

投稿

  • 評価
  • クリップ 0
  • VIEW 1,774

score 34

kotlinのScrollViewクラスで、スクロールが止まった時に処理が動くように実装することは可能でしょうか?
公式ドキュメントを調べても使えそうな関数は載っていませんでした。

https://developer.android.com/reference/kotlin/android/widget/ScrollView

やりたいことは、スクロールが止まった時の位置の取得です。

何か良い方法があれば教えていただきたいです。

よろしくお願いします。

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

質問への追記・修正、ベストアンサー選択の依頼

  • jimbe

    2019/10/19 13:13

    ご存じと思いますが, 人の目には動き続けているように見えて, 機械的には「少し動く←→止まる」を繰り返しています. なにをもって「スクロールが止まった」と判断するのでしょうか.

    キャンセル

  • takepon0209

    2019/10/19 13:51

    そうなんですね、知りませんでした。
    人の目から見て止まった時に処理を行いたいのですが、
    それは難しいということでしょうか?

    キャンセル

  • jimbe

    2019/10/19 23:38

    難しいかどうかは仕様次第ですが,
    > 人の目から見て止まった時
    という言われ方では, まだ抽象的です.
    例えば...移動する場合に 10 秒おきに 1 mm 進む生物が居たとします. どのような場合に「この生物は止まった(=移動していない)」と言えるでしょうか.

    キャンセル

  • takepon0209

    2019/10/21 16:29

    なるほど、そうですね。
    止まってから1秒動きがなかったらなど、止まってからの経過秒数で判定したいです。

    キャンセル

回答 2

+2

jimbeさんの回答だとOnScrollChangeListenerを使用していてAndroid M(APIレベル23)以降でしか動かないので、それ以前のバージョンでも動作させたい場合は以下のようにScrollViewを拡張したクラスを実装して代わりに利用する方法があります。

class StopAwareScrollView(...) : ScrollView(...) {
    private var debounceTask: Runnable? = null

    var onStopListener: (Int, Int) -> Unit = { _, _ -> }

    override fun onScrollChanged(l: Int, t: Int, oldl: Int, oldt: Int) {
        super.onScrollChanged(l, t, oldl, oldt)

        removeCallbacks(debounceTask)
        debounceTask = Runnable { onStopListener(l, t) }
        postDelayed(debounceTask, 1000L)
    }

    override fun onDetachedFromWindow() {
        super.onDetachedFromWindow()
        removeCallbacks(debounceTask)
    }
}
  • イベントを受け取る側
scrollView.onStopListener = { x, y ->
    // do something...
}

なお、こちらの実装ではThreadではなくHandlerを利用しているので、リスナーはメインスレッド上で呼ばれます。

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2019/10/22 15:57

    ありがとうございました!

    キャンセル

  • 2019/10/22 18:34

    私にも kotlin の勉強になりました. ありがとうございます.

    キャンセル

checkベストアンサー

+1

時間の監視でよろしければ, スクロールしたらリセットするタイマーを作り, そのタイマーが1秒経過したら「停止中」と判断しては如何でしょうか.

kotlin が使えないため java で失礼します.
非常にざっくりです.

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="com.teratail.q218060.MainActivity">

    <ScrollView
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:id="@+id/scrollView">

        <TextView
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:id="@+id/textView"/>
    </ScrollView>
</androidx.constraintlayout.widget.ConstraintLayout>


MainActivity.java

package com.teratail.q218060;

import androidx.appcompat.app.AppCompatActivity;

import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.ScrollView;
import android.widget.TextView;

public class MainActivity extends AppCompatActivity {
  private ScrollStopDetector scrollStopDetector;
  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    TextView textView = findViewById(R.id.textView);
    for(int i=1; i<=50; i++) textView.append(i+"\n");

    ScrollView scrollView = findViewById(R.id.scrollView);
    scrollStopDetector = new ScrollStopDetector(scrollView, 1000, new ScrollStopDetector.Listener() {
      @Override
      public void stopped(View v) {
        Log.d("MainActivity", "Scroll STOP");
      }
    });
    scrollView.setOnScrollChangeListener(scrollStopDetector);

    scrollStopDetector.start();
  }
  @Override
  protected void onDestroy() {
    super.onDestroy();
    scrollStopDetector.finish();
  }
}


ScrollStopDetector.java

package com.teratail.q218060;

import android.view.View;
import android.widget.ScrollView;

class ScrollStopDetector implements View.OnScrollChangeListener {
  interface Listener {
    void stopped(View v);
  }

  private class Sleeper implements Runnable {
    @Override
    public void run() {
      while(!finish) {
        try {
          Thread.sleep(millis); //[ms]
          listener.stopped(view);
        } catch(InterruptedException ignore) {
        }
      }
    }
  }

  private ScrollView view;
  private long millis;
  private Listener listener;
  private Thread thread;
  private boolean finish;

  ScrollStopDetector(ScrollView view, long millis, Listener listener) {
    this.view = view;
    this.millis = millis;
    this.listener = listener;
  }
  void start() {
    thread = new Thread(new Sleeper());
    thread.start();
  }
  void finish() {
    finish = true;
  }

  @Override
  public void onScrollChange(View v, int scrollX, int scrollY, int oldScrollX, int oldScrollY) {
    if(thread != null) thread.interrupt();
  }
}

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2019/10/22 15:57

    ありがとうございます!
    無事解決しました。

    キャンセル

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 87.37%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る