0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 3 years have passed since last update.

JavaFXでダイアログの入力チェック制御をちゃんと考えてみる

Posted at

この記事について

将来的に書く予定の「JavaFX で DynamoDB Viewer作ってみた」記事の1ステップ。
結構大きな話になると思うので、少しずつ技術ポイント毎に記事を書いて、ある一定程度の要件を満たせた段階で前述まとめ記事書く予定。

第一回記事:DynamoDBの情報を読み込んでJavaFXで表示してみる
第二回記事:JavaFXで動的にテーブル列を設定する
第三回記事:AWS java SDKでDynamoDBテーブル情報を取得してみる
第四回記事:JavaFX の TableView の選択範囲をクリップボードにコピーする。
第五回記事:JavaFX でコンポーネント作って動的生成してみる
第六回記事:DynamoDBのデータ型をjava SDKから把握してみる。
第七回記事:JavaFXでジェネリック使って入力ダイアログ作成してみる
第八回記事:JavaFXで各種サイズ制御
第九回記事:JavaFXのDialogで入力値チェック制御
第十回記事:DynamoDBでPartiQLをjavaSDKで実行してみる。

※これまでの記事が基本になってます。メソッドなど細かい部分で再説明していない部分があります。不明点などありましたらコメントなど頂けたら対応しようと思います。

今回の追加機能

  • キャンセルの時にも走っていたチェックなどを整理して、OKボタンの時のみチェックやDB更新確認を実行
  • 読み込み時のダイアログ表示がちゃんとしてなかったので制御
  • PartiQLでは、テーブル名にハイフン( - )があるとエラー(Unexpected from source)になるのでテーブル名をダブルコーテーションで括るように

現在の進捗

image.png

この時点でのソース(github)

状況

第九回記事:JavaFXのDialogで入力値チェック制御で、入力値チェック制御を行ったが、キャンセル時にもチェックが走ってしまい煩わしい。いいかげんちゃんと制御したい。

JavaFXの標準ライブラリのソースを見てみる

重要なイベントを見て、制御方法を考える

javafx.scene.control.Dialog

closeイベント

Dialog.java(※省略部分あり&説明コメント追加)

    public final void close() {
        if (isClosing) return;
        isClosing = true;

        // ダイアログの結果取得処理が走る。結果がnullなら後続処理でcancel扱いになる。
        final R result = getResult();

        // nullにした時、Cancelボタンが無いモードなら閉じない
        if (result == null && ! dialog.requestPermissionToClose(this)) {
            isClosing = false;
            return;
        }

        // 結果がnullの時、ダミーの結果をセットして再度closeする
        if (result == null) {
            // 中略

            // キャンセル処理を強制的に実行する
            impl_setResultAndClose(cancelButton, false);
        }

        // DIALOG_HIDINGイベント実行
        Event.fireEvent(this, new DialogEvent(this, DialogEvent.DIALOG_HIDING));

        // DIALOG_CLOSE_REQUESTイベント実行
        DialogEvent closeRequestEvent = new DialogEvent(this, DialogEvent.DIALOG_CLOSE_REQUEST);
        Event.fireEvent(this, closeRequestEvent);

        // DIALOG_CLOSE_REQUESTイベントの中で、isConsumedされたら閉じない
        if (closeRequestEvent.isConsumed()) {
            isClosing = false;
            return;
        }

        dialog.close();

        Event.fireEvent(this, new DialogEvent(this, DialogEvent.DIALOG_HIDDEN));

        isClosing = false;
    }

getResult

    /**
     * A property representing what has been returned from the dialog. A result
     * is generated through the {@link #resultConverterProperty() result converter},
     * which is intended to convert from the {@link ButtonType} that the user
     * clicked on into a value of type R. Refer to the {@link Dialog} class
     * JavaDoc for more details.
     */
    public final ObjectProperty<R> resultProperty() {
        return resultProperty;
    }

    public final R getResult() {
        return resultProperty().get();
    }

    public final void setResult(R value) {
        this.resultProperty().set(value);
    }

なになに、resultConverterProperty メソッドを通して作成される。この関数はクリックされたボタンによって変換される、と。

getResultConverter

    /**
     * API to convert the {@link ButtonType} that the user clicked on into a
     * result that can be returned via the {@link #resultProperty() result}
     * property. This is necessary as {@link ButtonType} represents the visual
     * button within the dialog, and do not know how to map themselves to a valid
     * result - that is a requirement of the dialog implementation by making use
     * of the result converter. In some cases, the result type of a Dialog
     * subclass is ButtonType (which means that the result converter can be null),
     * but in some cases (where the result type, R, is not ButtonType or Void),
     * this callback must be specified.
     */
    public final ObjectProperty<Callback<ButtonType, R>> resultConverterProperty() {
        return resultConverterProperty;
    }

    public final Callback<ButtonType, R> getResultConverter() {
        return resultConverterProperty().get();
    }

    public final void setResultConverter(Callback<ButtonType, R> value) {
        this.resultConverterProperty().set(value);
    }

渡されたButtonTypeによって結果を返すCallbackをセットせよ、という事ですね。
すなわち、現時点のresultConverterは制御不十分、と。

設計

以下のポイントがある。

  • OKした時の入力値チェックして駄目なら閉じない
  • キャンセルした時は入力値状態によらず閉じる
  • 大元ダイアログでOKする時はDB更新確認

これらの判断には「押されたボタン」「入力されている値が正常か」の情報が必要。

嫌なポイントが一つ。resultConverter でnullを返したとしても、cancelボタンが存在すると、cancel処理が強制的に走ってしまい、OKボタンを押したのに最終的にキャンセルボタンを押した時と同じようになるという(これはDialogクラスの設計ミスだと思う)。

resultConverter内修正

  • OKボタンの時のみ値チェックをする
  • その時、押されたボタンを保持しておく(前述「押されたボタン」情報)
  • 同じく、チェック結果を保持しておく(前述「入力されている値が正常か」情報)
  • 値チェックがFalseの時は、nullでなく、前述ポイント対策として、ダミーの空の値を返しておく
AbsDynamoDbInputDialog.java
	private ButtonData buttonData;

	// ・・中略・・

	public AbsDynamoDbInputDialog(R dynamoDbRecord, String dialogTitle) {
		// ・・中略・・
		setResultConverter((dialogButton) -> {
			buttonData = dialogButton == null ? null : dialogButton.getButtonData();
			if (buttonData  == ButtonData.OK_DONE) {
				isValidData = isFinalValidationOk();
				return isValidData ? getEditedDynamoDbRecord() : getEmptyAttr();
			}
			return null;
		});
		// ・・中略・・
	}

	// ・・中略・・

	abstract boolean isFinalValidationOk();
	abstract R getEditedDynamoDbRecord();
	abstract R getEmptyAttr();

	// ・・中略・・

	protected ButtonData getButtonData() {
		return buttonData;
	}

CloseRequestで終了制御

  • Cancelボタンの時(OKボタンでない時)は終了OK
  • OKボタンの時、値チェックが駄目なら終了NG(dialogEventをconsume)
AbsDynamoDbInputDialog.java
	// コンストラクタ内
	this.setOnCloseRequest(this::doFinalConfirmation);

	// ・・中略・・

	protected void doFinalConfirmation(DialogEvent dialogEvent) {
		if (buttonData != ButtonData.OK_DONE) {
			return;
		}
		if (!isValidData) {
			dialogEvent.consume();
		}
	}

大元ダイアログで、確認処理を入れる

doFinalConfirmationを継承して確認処理追加

  • 継承元で、閉じるのをキャンセルされて(isConsumed)いたらスルー
  • 押されたボタンがOKの時だけチェック
DynamoDbRecordInputDialog.java

	// ・・中略・・
	@Override
	protected void doFinalConfirmation(DialogEvent dialogEvent) {
		super.doFinalConfirmation(dialogEvent);
		if (!dialogEvent.isConsumed() && getButtonData() == ButtonData.OK_DONE) {
			Dialog<ButtonType> dialog = new Dialog<>();
			dialog.setContentText("This DynamoDB record will be update. Is it OK?");
			dialog.getDialogPane().getButtonTypes().addAll(ButtonType.YES, ButtonType.NO);
			dialog.showAndWait().ifPresent(buttonType -> {
				if (buttonType != ButtonType.YES) {
					dialogEvent.consume();
				}
			});
		}
	}

その他修正

  • 読み込みダイアログがちゃんと制御できてなかったので、
    JavaFX show loading dialog for longer operationsを参考に、Taskを使った制御に変更
  • PartiQLのテーブル名をダブルコーテーションで括るように

次回予定

csv、json形式などでのファイル出力

参考にさせて頂いたページ

JavaFX show loading dialog for longer operations
dynamodb PartiQL SELECT query returns ValidationException: Unexpected from source

0
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?