4

更新: 解決済み、コード付き

コードについては、以下の回答を参照してください...

元の投稿

Tundeyが私の最後の質問への回答で指摘したように、Windows フォーム コントロールに関するほぼすべてを ApplicationSettings に非常に簡単にバインドできます。フォームサイズでこれを行う方法は本当にありませんか? このチュートリアルでは、ウィンドウが最大化または最小化されている場合にサイズの代わりに RestoreBounds を保存できるように、Size を明示的に処理する必要があると述べています。ただし、次のようなプロパティを使用できることを望みました。

public Size RestoreSize
{
    get
    {
        if (this.WindowState == FormWindowState.Normal)
        {
            return this.Size;
        }
        else
        {
            return this.RestoreBounds.Size;
        }
    }
    set
    {
        ...
    }
}

しかし、デザイナーでこれをバインドする方法がわかりません (PropertyBinding リストにサイズが特にありません)。

4

5 に答える 5

12

私は最終的に、これを完全に解決する Form サブクラスを思い付きました。使用するには:

  1. Form ではなく RestorableForm から継承します。
  2. (ApplicationSettings) -> (PropertyBinding) でバインディングを WindowRestoreState に追加します。
  3. ウィンドウが閉じようとしているときに、Properties.Settings.Default.Save() を呼び出します。

ウィンドウの位置と状態がセッション間で記憶されるようになりました。以下の他のポスターからの提案に従って、ウィンドウ自体を復元するときにウィンドウが利用可能なディスプレイにうまく収まるようにする関数 ConstrainToScreen を含めました。

コード

// Consider this code public domain. If you want, you can even tell
// your boss, attractive women, or the other guy in your cube that
// you wrote it. Enjoy!

using System;
using System.Windows.Forms;
using System.ComponentModel;
using System.Drawing;

namespace Utilities
{
    public class RestorableForm : Form, INotifyPropertyChanged
    {
        // We invoke this event when the binding needs to be updated.
        public event PropertyChangedEventHandler PropertyChanged;

        // This stores the last window position and state
        private WindowRestoreStateInfo windowRestoreState;

        // Now we define the property that we will bind to our settings.
        [Browsable(false)]        // Don't show it in the Properties list
        [SettingsBindable(true)]  // But do enable binding to settings
        public WindowRestoreStateInfo WindowRestoreState
        {
            get { return windowRestoreState; }
            set
            {
                windowRestoreState = value;
                if (PropertyChanged != null)
                {
                    // If anybody's listening, let them know the
                    // binding needs to be updated:
                    PropertyChanged(this,
                        new PropertyChangedEventArgs("WindowRestoreState"));
                }
            }
        }

        protected override void OnClosing(CancelEventArgs e)
        {
            WindowRestoreState = new WindowRestoreStateInfo();
            WindowRestoreState.Bounds
                = WindowState == FormWindowState.Normal ?
                  Bounds : RestoreBounds;
            WindowRestoreState.WindowState = WindowState;

            base.OnClosing(e);
        }

        protected override void OnLoad(EventArgs e)
        {
            base.OnLoad(e);

            if (WindowRestoreState != null)
            {
                Bounds = ConstrainToScreen(WindowRestoreState.Bounds);
                WindowState = WindowRestoreState.WindowState;
            }
        }

        // This helper class stores both position and state.
        // That way, we only have to set one binding.
        public class WindowRestoreStateInfo
        {
            Rectangle bounds;
            public Rectangle Bounds
            {
                get { return bounds; }
                set { bounds = value; }
            }

            FormWindowState windowState;
            public FormWindowState WindowState
            {
                get { return windowState; }
                set { windowState = value; }
            }
        }

        private Rectangle ConstrainToScreen(Rectangle bounds)
        {
            Screen screen = Screen.FromRectangle(WindowRestoreState.Bounds);
            Rectangle workingArea = screen.WorkingArea;

            int width = Math.Min(bounds.Width, workingArea.Width);
            int height = Math.Min(bounds.Height, workingArea.Height);

            // mmm....minimax
            int left = Math.Min(workingArea.Right - width,
                                Math.Max(bounds.Left, workingArea.Left));
            int top = Math.Min(workingArea.Bottom - height,
                                Math.Max(bounds.Top, workingArea.Top));

            return new Rectangle(left, top, width, height);
        }
    }
}

設定バインディングのリファレンス

于 2008-08-20T23:06:42.563 に答える
7

Form.Sizeプロパティが設定バインディングUIで使用できない理由は、このプロパティがDesignerSerializationVisibility.Hiddenとマークされているためです。これは、設計者がシリアル化する方法を知らないことを意味します。ましてや、データバインディングを生成することもできません。代わりに、Form.ClientSizeプロパティがシリアル化されます。

LocationClientSizeをバインドして賢くしようとすると、別の問題が発生します。フォームのサイズを左端または上端から変更しようとすると、奇妙な動作が見られます。これは、相互に影響を与えるプロパティセットのコンテキストで双方向データバインディングが機能する方法に明らかに関連しています。LocationClientSizeの両方が、最終的に共通のメソッドSetBoundsCore()を呼び出します。

また、 LocationSizeなどのプロパティへのデータバインディングは効率的ではありません。ユーザーがフォームを移動またはサイズ変更するたびに、Windowsは何百ものメッセージをフォームに送信し、フォームを閉じる前に最後の位置とサイズを保存するだけで、データバインディングロジックに多くの処理を実行させます。

これは私がしていることの非常に単純化されたバージョンです:

private void MyForm_FormClosing(object sender, FormClosingEventArgs e)
{
    Properties.Settings.Default.MyState = this.WindowState;
    if (this.WindowState == FormWindowState.Normal)
    {
       Properties.Settings.Default.MySize = this.Size;
       Properties.Settings.Default.MyLoc = this.Location;
    }
    else
    {
       Properties.Settings.Default.MySize = this.RestoreBounds.Size;
       Properties.Settings.Default.MyLoc = this.RestoreBounds.Location;
    }
    Properties.Settings.Default.Save();
}

private void MyForm_Load(object sender, EventArgs e)
{
    this.Size = Properties.Settings.Default.MySize;
    this.Location = Properties.Settings.Default.MyLoc;
    this.WindowState = Properties.Settings.Default.MyState;
} 

なぜこれは非常に単純化されたバージョンなのですか?これを適切に行うのは見た目よりもかなり難しいからです:-)

于 2008-11-04T02:59:21.203 に答える
2

サイズバインディングが許可されていないと思う理由の 1 つは、セッション間で画面が変わる可能性があるためです。

解像度が低下したときにサイズを元に戻すと、タイトル バーが画面の制限を超える可能性があります。

また、アプリを次に実行したときにモニターが使用できなくなる可能性がある複数のモニターのセットアップにも注意する必要があります。

于 2008-08-20T19:52:06.713 に答える
1

フォームのサイズを AppSettings に直接バインドする方法はありませんが、独自の値を追加してロード時にサイズを変更できます。

これが一般的な機能である場合は、Form をサブクラス化し、フォームのサイズ設定について App.Config を自動的に調べるようにすることをお勧めします。

(または、独自のファイルをロールすることもできます..Xml ファイル「formname.settings.xml」または何かを照会するように取得しますか? - 大声で考えてください!)..

これが私が持っていたものです(非常に大まかで、エラーチェックなどはありません)。

App.Config

<?xml version="1.0" encoding="utf-8" ?>
<configuration>
    <appSettings>
        <add key ="FormHeight" value="500" />
        <add key ="FormWidth" value="200"/>
    </appSettings>
</configuration>

フォームコード

    private void Form1_Load(object sender, EventArgs e)
    {
        string height = ConfigurationManager.AppSettings["FormHeight"];
        int h = int.Parse(height);
        string width = ConfigurationManager.AppSettings["FormWidth"];
        int w = int.Parse(width);
        this.Size = new Size(h, w);
    }
于 2008-08-20T19:41:37.097 に答える
1

ロブ・クーパーの答えに同意します。しかし、マーティンは非常に良い点を指摘していると思います。ユーザーがアプリケーションを開いて、アプリが画面外にあるようなものはありません!

したがって、実際には、フォームのサイズを設定する前に、両方の回答を組み合わせて、現在の画面サイズを念頭に置く必要があります。

于 2008-08-20T20:04:16.903 に答える