3

メインフォームから次のように子フォームを開きます。

var cf = new ChildForm { Owner = this };
cf.Show();

次に、子フォームは別のスレッドで描画を行います。

ユーザーがメインフォームを閉じようとすると(子フォームが開いている場合)、FormClosing最初にChildFormイベントが発生し、次にメインフォームで同じイベントが発生します。イベントが発生するFormClosingと、子フォームは描画スレッドを停止します。

保存されていないデータが含まれている場合、ユーザーはメインフォームを閉じようとする可能性があります。次に、メインフォームのイベントハンドラーによって、「データが保存されていません。閉じるのをキャンセルしますか? 」という警告が表示されます。FormClosingその後、保存をキャンセルできます(つまり、CancelフラグはFormClosingEventArgsメインフォームのFormClosingイベントハンドラーによってオブジェクトに設定されます)。

ただし、その時点で、子フォームのFormClosingイベントはすでに発生しており、描画スレッドは停止します。子フォームは、(何も起こらなかったかのように)描画を続行する必要があることを認識していません。

子フォームからFormClosing、メインフォームによってイベントがキャンセルされたことを検出することはできますか?ユーザーがメインフォームにデータを保存するように求められている間も、スレッドの再描画を停止したいと思います。

4

3 に答える 3

2

インターフェイスに基づいたソリューションを提供します。この方法は、アプリケーションを閉じることができるかどうかを管理するための統一された方法を簡単に持つことができます。次の実装では、親フォームが子ウィンドウに閉じる準備ができているかどうかを尋ね、子は実行する必要のあるアクションを実行し、メインウィンドウに応答します。

私がインターフェースを持っているとしましょうIManagedForm

interface IManagedForm
{
    bool CanIBeClosed(Object someParams);
}

両方の形式(Form1およびChildForm)がそれを実装します。

この例では、を次のようにインスタンス化していることに注意してくださいChildForm

ChildForm cf = new ChildForm() { Owner = this, Name = "ChildForm" };
cf.Show();

ここでは、最初に次の方法でインターフェイスを実装しますForm1

private void Form1_FormClosing(object sender, FormClosingEventArgs e)
{
    object someArgsInterestingForTheMethod = new object();

    e.Cancel = !((IManagedForm)this).CanIBeClosed(someArgsInterestingForTheMethod);
}

// Ask the ChildForm it is done. If not the user should not leave the application.
public bool CanIBeClosed(object someParams)
{
    bool isOKforClosing = true;

    var cf = this.Controls["ChildForm"] as IManagedForm;

    if (cf != null)
    {
        isOKforClosing = cf.CanIBeClosed(someParams);
        if (!isOKforClosing)
        {
            MessageBox.Show("ChildForm does not allow me to close.", "Form1", MessageBoxButtons.OK);
        }
    }
    return isOKforClosing;
}

そして最後ChildFormに、インターフェースの実装は次のようになります。

private void ChildForm_FormClosing(object sender, FormClosingEventArgs e)
{
    object someArgsInterestingForTheMethod = new object();

    e.Cancel = !((IManagedForm)this).CanIBeClosed(someArgsInterestingForTheMethod);
}

public bool CanIBeClosed(object someParams)
{
    // This flag would control if this window has not pending changes.
    bool meetConditions = ValidateClosingConditions(someParams);
    // If there were pending changes, but the user decided to not discard
    // them an proceed saving, this flag says to the parent that this form
    // is done, therefore is ready to be closed.
    bool iAmReadyToBeClosed = true;

    // There are unsaved changed. Ask the user what to do.
    if (!meetConditions)
    {
        // YES => OK Save pending changes and exit.
        // NO => Do not save pending changes and exit.
        // CANCEL => Cancel closing, just do nothing.
        switch (MessageBox.Show("Save changes before exit?", "MyChildForm", MessageBoxButtons.YesNoCancel, MessageBoxIcon.Question))
        {
            case DialogResult.Yes:
                // Store data and leave...
                iAmReadyToBeClosed = true;
                break;
            case DialogResult.No:
                // Do not store data, just leave...
                iAmReadyToBeClosed = true;
                break;
            case DialogResult.Cancel:
                // Do not leave...
                iAmReadyToBeClosed = false;
                break;
        }
    }
    return iAmReadyToBeClosed;
}

// This is just a dummy method just for testing
public bool ValidateClosingConditions(object someParams)
{
    Random rnd = new Random();
    return ((rnd.Next(10) % 2) == 0);
}

それが十分に明確であることを願っています。

于 2012-07-03T08:49:49.347 に答える
1

はい/いいえ/キャンセルのいずれかの乾燥ロジックは、いくつかのセトラルクラスに移動します。
そしてFormClosingのどちらかが同じ関数を呼び出します。

当然のことながら、優れたユーザーエクスペリエンスを実現するには、関数が両方の関数から呼び出された場合に1回だけ実行されるように管理する必要がありFormClosingます。

于 2012-07-03T06:38:24.093 に答える
0

MSDNは、FormClosingイベントを、フォームが閉じる前に呼び出されるイベントとして説明しています。さらに、MDIの場合、親フォームのFormClosingイベントが呼び出される前に、すべての子フォームのFormClosingイベントが呼び出されます。

つまり、メインフォームで閉じるボタンを押すと、すべての子フォームに対してFormClosingイベントが発生します。したがって、子フォームは、閉じる準備ができているかどうかを判断し、それに応じてcancelプロパティを設定する必要があります。メインフォームでイベントが発生した場合、すでにキャンセルするように設定されているはずです。

メインフォームでクローズイベントが発生した場合でも、メインフォームは子フォームを決定するべきではありません。

さらに読むために、

MSDNでのForm.FormClosingイベントの説明

キャンセルプロパティの使用方法の説明

于 2012-07-03T07:06:39.873 に答える