ScrollPaneの縦スクロール位置をプログラム側で変更したい
- 評価
- クリップ 0
- VIEW 3,741
JavaFXのScrollPaneについて質問です。
画面上に表示しているScrollPaneの縦スクロール位置を、ユーザの操作ではなく、プログラム側で変更するには、どうすれば良いでしょうか?
(※説明の為に、変数名は実際のプログラムから一部変更しています。)
私がJavaFXで現在作成しているアプリケーションでは、縦スクロールのScrollPane(scroll)の中にFlowPane(listField)があります。
ユーザの操作によって、このlistFieldの中にGridPaneの表(list)を表示しています。
しかしこれだと、新しいlistをlistFieldの中に表示する度に、毎回scrollが一番上にスクロールした状態になってしまいます。
その為、scrollの縦スクロール位置を一旦変数vValueに入れ、それからlistFieldの中に新しく作ったlistを表示し、最後にscrollの縦スクロール位置をvValueに合わせれば良いと考えました。
下記プログラムの通り、getVvalueメソッドでscrollの縦スクロール位置は取得出来ました。
double vValue=scroll.getVvalue();
listField.getChildren().clear(); //listField内全削除
GridPane list=createList(); //list生成
listField.getChildren().addAll(list); //listFieldにlistを表示
scroll.setVvalue(vValue); //新listのスクロール位置を旧listと同じ位置に変える
しかし、setVvalueメソッドを使っても、scrollの縦スクロール位置を画面上で移動させる事が出来ませんでした。
確認してみた所、setVvalueメソッドの後は、listの縦スクロール位置の数値自体は変化していました。
調べた所、恐らく同様の質問をされていると思われる下記ページが見付かりました。
参考サイト
英語な事もあり詳しくは分かりませんでしたが、ScrollPaneのlayoutメソッドというのを使ってから、setVvalueメソッドで縦スクロール数値を指定すべきだという意味だと解釈しました。
double vValue=scroll.getVvalue();
listField.getChildren().clear(); //listField内全削除
GridPane list=createList(); //list生成
listField.getChildren().addAll(list); //listFieldにlistを表示
scroll.layout(); //?????
scroll.setVvalue(vValue); //新listのスクロール位置を旧listと同じ位置に変える
しかしこれでも、scrollの縦スクロール位置が画面上で移動する事はありませんでした。
どの様にすれば縦スクロール位置を変更出来るでしょうか?
-
気になる質問をクリップする
クリップした質問は、後からいつでもマイページで確認できます。
またクリップした質問に回答があった際、通知やメールを受け取ることができます。
クリップを取り消します
-
良い質問の評価を上げる
以下のような質問は評価を上げましょう
- 質問内容が明確
- 自分も答えを知りたい
- 質問者以外のユーザにも役立つ
評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。
質問の評価を上げたことを取り消します
-
評価を下げられる数の上限に達しました
評価を下げることができません
- 1日5回まで評価を下げられます
- 1日に1ユーザに対して2回まで評価を下げられます
質問の評価を下げる
teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。
- プログラミングに関係のない質問
- やってほしいことだけを記載した丸投げの質問
- 問題・課題が含まれていない質問
- 意図的に内容が抹消された質問
- 過去に投稿した質問と同じ内容の質問
- 広告と受け取られるような投稿
評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。
質問の評価を下げたことを取り消します
この機能は開放されていません
評価を下げる条件を満たしてません
質問の評価を下げる機能の利用条件
この機能を利用するためには、以下の事項を行う必要があります。
- 質問回答など一定の行動
-
メールアドレスの認証
メールアドレスの認証
-
質問評価に関するヘルプページの閲覧
質問評価に関するヘルプページの閲覧
0
ずいぶん時間が経ってしまいましたが・・・
残念ながら今現在「こうすべき」という適切な対処はわからないのですが、ヒントになるかも知れない情報をコメントしてみます。
原因(のヒント)
参照しておられるstackoverflowのページは質問者さんがやろうとしていることとほぼ同様のことをしていると思います。そちらのコードで試してみました。自分の環境(JavaSE 9.0.4, Windows 10)ではstackoverflowの回答にあるlayoutをやってもNGでした。
ふと思いついてvvaluePropertyの変化をChangeListenerで調べたところ元のvValueの値を再設定した後、しばらくしてvvaluePropertyの値が0.0にリセットされていることがわかりました。0.0にリセットされた際のスタックトレースをみるとpulse(JavaFXランタイムが定期的に行う処理)の中でfocusCleanupという処理が行われており、そこから巡り巡ってScrollPaneSkin(ScrollPaneの外観/内部的な振る舞いを制御するSkinクラス)のupdatePosYというメソッドでvvaluePropertyの値が再設定(再調整?)されており、そこで0.0が設定されてしまうのが原因のようでした。
本来、pulseのようなJavaFXランタイムの内部で行われる定期処理のタイミングを意識して制御するようなことはアプリケーションプログラマーが考慮する必要がないように設計されているべきな気がするのですが、実際pulseの中でvvaluePropertyの値が調整されてしまう事実がある以上は「どうすべきなのか」自分にはわかりませんでした。
対処(のヒント?)
対処というよりイマイチな回避策というべきですが、「pulseが1回以上実行された(であろう)後にvvaluePropertyの値を設定する」という方法を試してみました。
public class Main extends Application {
...
boolean timerRestart = false;
AnimationTimer timer = new AnimationTimer() {
long start;
@Override
public void handle(long now) {
if (timerRestart) {
start = now;
timerRestart = false;
return;
}
if (now - start > 60_000_000) {
System.out.println("now recover original vValue");
scroll.setVvalue(vValue);
timer.stop();
}
}
};
public void returnToResults() {
resultsBox.getChildren().clear();
resultsBox.getChildren().addAll(results); // (A)
returnToResults.setDisable(true);
timerRestart = true;
timer.start();
}
...
}
おおまかな動作は以下の通りです。
(1) scrollPaneの内容を(A)で入れ替えた際にAnimationTimerをstart
(2) AnimationTimerをstart後適当な時間が経過するまで待つ(上の例ではとりあえず60ms)
pulseはJavaFXのアプリケーションスレッドをよほど高負荷にしない限り1秒間に数十回は起動されると思いますので、とりあえず適当に60msとしましたがもっと短い間隔でもよいはずと思います。
(3) pulseが実行されたであろう時間経過の後vvaluePropetyを元の値に設定し、AnimationTimerをstopする。
なんとも不格好な回避策ですが、やってみたところ一応期待通りの動作になりました。
投稿
-
回答の評価を上げる
以下のような回答は評価を上げましょう
- 正しい回答
- わかりやすい回答
- ためになる回答
評価が高い回答ほどページの上位に表示されます。
-
回答の評価を下げる
下記のような回答は推奨されていません。
- 間違っている回答
- 質問の回答になっていない投稿
- スパムや攻撃的な表現を用いた投稿
評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。
15分調べてもわからないことは、teratailで質問しよう!
- ただいまの回答率 88.23%
- 質問をまとめることで、思考を整理して素早く解決
- テンプレート機能で、簡単に質問をまとめられる