4

更新:已解决,带有代码

我得到了它的工作,请参阅下面的代码答案...

原帖

正如 Tundey 在他对我最后一个问题的回答中指出的那样,您几乎可以毫不费力地将有关 Windows 窗体控件的所有内容绑定到 ApplicationSettings。那么表格大小真的没有办法做到这一点吗?本教程说您需要显式处理 Size 以便在窗口最大化或最小化时保存 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. 继承自 RestorableForm 而不是 Form。
  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变得聪明,您会看到另一个问题。当您尝试从左边缘或上边缘调整表单大小时,您会看到奇怪的行为。这显然与双向数据绑定在相互影响的属性集的上下文中的工作方式有关。Location和ClientSize最终都会调用一个通用方法SetBoundsCore ()

此外,数据绑定到诸如位置大小等属性的效率并不高。每次用户移动或调整窗体大小时,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

我认为不允许大小绑定的原因之一是屏幕可能会在会话之间发生变化。

在分辨率降低时重新加载大小可能会导致标题栏超出屏幕的限制。

您还需要警惕多个监视器设置,当您下次运行应用程序时,监视器可能不再可用。

于 2008-08-20T19:52:06.713 回答
1

好吧,我对此进行了快速尝试,您是对的,虽然无法将表单的大小直接绑定到 AppSettings,但您可以添加自己的值并更改加载时的大小。

我可能会建议,如果这是一个常见功能,您可以将 Form 子类化并使其自动探测 App.Config 的表单大小设置。

(或者你可以滚动你自己的文件..让它查询一个 Xml 文件“formname.settings.xml”或其他东西? - 大声思考!)..

这是我所拥有的(非常粗糙,没有错误检查等)。

应用程序配置

<?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

我同意 Rob Cooper 的回答。但我认为马丁提出了一个很好的观点。没有什么比让用户打开您的应用程序并且应用程序不在屏幕上更有趣了!

所以实际上,在设置表单大小之前,您需要结合这两个答案并记住当前的屏幕尺寸。

于 2008-08-20T20:04:16.903 回答