Почему вы не можете привязать размер окна к ApplicationSettings? - PullRequest
2 голосов
/ 20 августа 2008

Обновление: решено, с кодом

У меня получилось, см. Мой ответ ниже для кода ...

Оригинальный пост

Как Тандей указал в своем ответе на мой последний вопрос , вы можете довольно легко связать практически все, что касается управления окнами, с ApplicationSettings. Так неужели нет способа сделать это с размером формы? В этом руководстве говорится, что вам нужно явно обрабатывать размер, чтобы вы могли сохранять RestoreBounds вместо размера, если окно развернуто или свернуто. Тем не менее, я надеялся, что смогу просто использовать свойство как:

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

Но я не вижу способа связать это в конструкторе (размер заметно отсутствует в списке PropertyBinding).

Ответы [ 5 ]

11 голосов
/ 21 августа 2008

Я наконец-то придумал подкласс 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);
        }
    }
}

Настройки Привязки Ссылки

5 голосов
/ 04 ноября 2008

Причина, по которой свойство Form.Size недоступно в пользовательском интерфейсе привязки параметров, заключается в том, что это свойство помечено DesignerSerializationVisibility.Hidden . Это означает, что разработчик не знает, как его сериализовать, не говоря уже о создании привязки данных для него. Вместо этого свойство Form.ClientSize является тем, которое сериализуется.

Если вы попытаетесь получить ум, связав Местоположение и ClientSize , вы увидите еще одну проблему. Когда вы попытаетесь изменить размер вашей формы с левого или верхнего края, вы увидите странное поведение. По-видимому, это связано с тем, как двусторонняя привязка данных работает в контексте наборов свойств, которые взаимно влияют друг на друга. И Местоположение и 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;
} 

Почему это очень упрощенная версия? Потому что сделать это правильно намного сложнее , чем кажется: -)

1 голос
/ 20 августа 2008

Одна из причин, по которой я считаю, что привязка размера не разрешена, заключается в том, что экран может меняться между сеансами.

Загрузка размера назад при уменьшении разрешения может привести к тому, что строка заголовка выйдет за пределы экрана.

Вам также следует опасаться установки нескольких мониторов, когда мониторы могут быть недоступны при следующем запуске приложения.

1 голос
/ 21 августа 2008

Я согласен с ответом Роба Купера. Но я думаю, что Мартин делает очень хорошую мысль. Ничего подобного тому, чтобы пользователи открывали ваше приложение, а приложение находилось вне экрана!

Таким образом, на самом деле вам нужно объединить оба ответа и учитывать текущие размеры экрана перед установкой размера формы.

1 голос
/ 20 августа 2008

Ну, я быстро поиграл с этим, и вы правы, хотя нет прямого способа привязать размер формы к 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);
    }
...