C #: как заставить форму помнить ее Bounds и WindowState (принимая во внимание настройки двух мониторов) - PullRequest
9 голосов
/ 30 января 2009

Я создал класс, от которого форма может наследоваться, и он обрабатывает форму Location, Size и State. И это работает хорошо. За исключением одного:

Когда вы максимизируете приложение на экране, отличном от основного, местоположение и размер (до максимизации) сохраняются правильно, но когда оно максимизируется (в соответствии с предыдущим состоянием), оно максимизируется на моем главном мониторе. Когда я затем возвращаю его в нормальное состояние, он переходит на другой экран, где он был раньше. Когда я затем снова максимизирую его, он, конечно, максимизируется на правильном экране.

Итак, мой вопрос ... как я могу сделать форму, когда она развернута, помните, на каком экране она была развернута? И как мне восстановить это, когда форма открывается снова?


Вид полного решения проблемы

Я принял ответ, в котором был очень хороший совет о том, как, если на экране. Но это была только часть моей проблемы, поэтому вот мое решение:

под нагрузкой

  1. Сначала храните Bounds и WindowState из любого хранилища.
  2. Затем установите Bounds.
  3. Убедитесь, что Bounds видны либо Screen.AllScreens.Any(ø => ø.Bounds.IntersectsWith(Bounds)), либо MdiParent.Controls.OfType<MdiClient>().First().ClientRectangle.IntersectsWith(Bounds).
    • Если это не так, просто сделайте Location = new Point();.
  4. Затем установите состояние окна.

При закрытии

  1. Магазин WindowState.
  2. Если WindowState равно FormWindowState.Normal, сохраните Bounds, в противном случае сохраните RestoreBounds.

И это все! =) * * Тысяча пятьдесят-две

Пример кода

Итак, как подсказал Оливер , вот код. Это нужно конкретизировать, но это может быть использовано как начало для тех, кто хочет:

PersistentFormHandler

Заботится о хранении и извлечении данных где-либо.

public sealed class PersistentFormHandler
{
    /// <summary>The form identifier in storage.</summary>
    public string Name { get; private set; }


    /// <summary>Gets and sets the window state. (int instead of enum so that it can be in a BI layer, and not require a reference to WinForms)</summary>
    public int WindowState { get; set; }


    /// <summary>Gets and sets the window bounds. (X, Y, Width and Height)</summary>
    public Rectangle WindowBounds { get; set; }


    /// <summary>Dictionary for other values.</summary>
    private readonly Dictionary<string, Binary> otherValues;


    /// <summary>
    /// Instantiates new persistent form handler.
    /// </summary>
    /// <param name="windowType">The <see cref="Type.FullName"/> will be used as <see cref="Name"/>.</param>
    /// <param name="defaultWindowState">Default state of the window.</param>
    /// <param name="defaultWindowBounds">Default bounds of the window.</param>
    public PersistentFormHandler(Type windowType, int defaultWindowState, Rectangle defaultWindowBounds)
        : this(windowType, null, defaultWindowState, defaultWindowBounds) { }

    /// <summary>
    /// Instantiates new persistent form handler.
    /// </summary>
    /// <param name="windowType">The <see cref="Type.FullName"/> will be used as base <see cref="Name"/>.</param>
    /// <param name="id">Use this if you need to separate windows of same type. Will be appended to <see cref="Name"/>.</param>
    /// <param name="defaultWindowState">Default state of the window.</param>
    /// <param name="defaultWindowBounds">Default bounds of the window.</param>
    public PersistentFormHandler(Type windowType, string id, int defaultWindowState, Rectangle defaultWindowBounds)
    {
        Name = string.IsNullOrEmpty(id) 
            ? windowType.FullName 
            : windowType.FullName + ":" + id;

        WindowState = defaultWindowState;
        WindowBounds = defaultWindowBounds;

        otherValues = new Dictionary<string, Binary>();
    }


    /// <summary>
    /// Looks for previously stored values in database.
    /// </summary>
    /// <returns>False if no previously stored values were found.</returns>
    public bool Load()
    {
        // See Note 1
    }

    /// <summary>
    /// Stores all values in database
    /// </summary>
    public void Save()
    {
        // See Note 2
    }

    /// <summary>
    /// Adds the given <paramref key="value"/> to the collection of values that will be
    /// stored in database on <see cref="Save"/>.
    /// </summary>
    /// <typeparam key="T">Type of object.</typeparam>
    /// <param name="key">The key you want to use for this value.</param>
    /// <param name="value">The value to store.</param>
    public void Set<T>(string key, T value)
    {
        // Create memory stream
        using (var s = new MemoryStream())
        {
            // Serialize value into binary form
            var b = new BinaryFormatter();
            b.Serialize(s, value);

            // Store in dictionary
            otherValues[key] = new Binary(s.ToArray());
        }
    }

    /// <summary>
    /// Same as <see cref="Get{T}(string,T)"/>, but uses default(<typeparamref name="T"/>) as fallback value.
    /// </summary>
    /// <typeparam name="T">Type of object</typeparam>
    /// <param name="key">The key used on <see cref="Set{T}"/>.</param>
    /// <returns>The stored object, or the default(<typeparamref name="T"/>) object if something went wrong.</returns>
    public T Get<T>(string key)
    {
        return Get(key, default(T));
    }

    /// <summary>
    /// Gets the value identified by the given <paramref name="key"/>.
    /// </summary>
    /// <typeparam name="T">Type of object</typeparam>
    /// <param name="key">The key used on <see cref="Set{T}"/>.</param>
    /// <param name="fallback">Value to return if the given <paramref name="key"/> could not be found.
    /// In other words, if you haven't used <see cref="Set{T}"/> yet.</param>
    /// <returns>The stored object, or the <paramref name="fallback"/> object if something went wrong.</returns>
    public T Get<T>(string key, T fallback)
    {
        // If we have a value with this key
        if (otherValues.ContainsKey(key))
        {
            // Create memory stream and fill with binary version of value
            using (var s = new MemoryStream(otherValues[key].ToArray()))
            {
                try
                {
                    // Deserialize, cast and return.
                    var b = new BinaryFormatter();
                    return (T)b.Deserialize(s);
                }
                catch (InvalidCastException)
                {
                    // T is not what it should have been
                    // (Code changed perhaps?)
                }
                catch (SerializationException)
                {
                    // Something went wrong during Deserialization
                }
            }
        }

        // Else return fallback
        return fallback;
    }
}

Примечание 1: В методе загрузки вы должны искать ранее сохраненные WindowState, WindowBounds и другие значения. Мы используем SQL Server и имеем таблицу Window со столбцами для Id, Name, MachineName (для Environment.MachineName), UserId, WindowState, X, Y, Height, Width. Таким образом, для каждого окна у вас будет по одной строке с WindowState, X, Y, Height и Width для каждого пользователя и машины. Кроме того, у нас есть таблица WindowValues, в которой есть только внешний ключ для WindowId, столбец Key типа String и столбец Value типа Binary. Если есть вещи, которые не найдены, я просто оставляю вещи по умолчанию и возвращаю false.

Примечание 2: В методе сохранения вы затем, конечно, делаете обратное тому, что вы делаете в методе Load. Создание строк для Window и WindowValues, если они еще не существуют для текущего пользователя и компьютера.

PersistentFormBase

Этот класс использует предыдущий класс и образует удобный базовый класс для других форм.

// Should have been abstract, but that makes the the designer crash at the moment...
public class PersistentFormBase : Form
{
    private PersistentFormHandler PersistenceHandler { get; set; }

    private bool handlerReady;

    protected PersistentFormBase()
    {
        // Prevents designer from crashing
        if (LicenseManager.UsageMode != LicenseUsageMode.Designtime)
        {
            Load += persistentFormLoad;
            FormClosing += persistentFormFormClosing;
        }
    }

    protected event EventHandler<EventArgs> ValuesLoaded;
    protected event EventHandler<EventArgs> StoringValues;

    protected void StoreValue<T>(string key, T value)
    {
        if (!handlerReady)
            throw new InvalidOperationException();
        PersistenceHandler.Set(key, value);
    }

    protected T GetValue<T>(string key)
    {
        if (!handlerReady)
            throw new InvalidOperationException();
        return PersistenceHandler.Get<T>(key);
    }

    protected T GetValue<T>(string key, T fallback)
    {
        if (!handlerReady)
            throw new InvalidOperationException();
        return PersistenceHandler.Get(key, fallback);
    }

    private void persistentFormLoad(object sender, EventArgs e)
    {
        // Create PersistenceHandler and load values from it
        PersistenceHandler = new PersistentFormHandler(GetType(), (int) FormWindowState.Normal, Bounds);
        PersistenceHandler.Load();
        handlerReady = true;

        // Set size and location
        Bounds = PersistenceHandler.WindowBounds;

        // Check if we have an MdiParent
        if(MdiParent == null)
        {
            // If we don't, make sure we are on screen
            if (!Screen.AllScreens.Any(ø => ø.Bounds.IntersectsWith(Bounds)))
                Location = new Point();
        }
        else
        {
            // If we do, make sure we are visible within the MdiClient area
            var c = MdiParent.Controls.OfType<MdiClient>().FirstOrDefault();
            if(c != null && !c.ClientRectangle.IntersectsWith(Bounds))
                Location = new Point();
        }

        // Set state
        WindowState = Enum.IsDefined(typeof (FormWindowState), PersistenceHandler.WindowState) ? (FormWindowState) PersistenceHandler.WindowState : FormWindowState.Normal;

        // Notify that values are loaded and ready for getting.
        var handler = ValuesLoaded;
        if (handler != null)
            handler(this, EventArgs.Empty);
    }

    private void persistentFormFormClosing(object sender, FormClosingEventArgs e)
    {
        // Set common things
        PersistenceHandler.WindowState = (int) WindowState;
        PersistenceHandler.WindowBounds = WindowState == FormWindowState.Normal ? Bounds : RestoreBounds;

        // Notify that values will be stored now, so time to store values.
        var handler = StoringValues;
        if (handler != null)
            handler(this, EventArgs.Empty);

        // Save values
        PersistenceHandler.Save();
    }
}

И это во многом. Чтобы использовать его, форма будет просто наследоваться от PersistentFormBase. Это автоматически позаботится о границах и состоянии. Если что-то еще нужно сохранить, например, расстояние до разделителя, вы должны прослушивать события ValuesLoaded и StoringValues, а в них использовать методы GetValue и StoreValue.

Надеюсь, это может кому-то помочь! Пожалуйста, дайте мне знать, если это так. А также, пожалуйста, оставьте отзыв, если есть что-то, что, по вашему мнению, могло бы быть лучше или что-то в этом роде. Я хотел бы узнать =)

Ответы [ 4 ]

4 голосов
/ 30 января 2009

Нет встроенного способа сделать это - вам придется написать логику самостоятельно. Одна из причин этого заключается в том, что вам нужно решить, как обрабатывать случай, когда монитор, на котором в последний раз отображалось окно, больше не доступен. Это может быть довольно распространено, например, для ноутбуков и проекторов. Класс Screen имеет некоторые полезные функции, чтобы помочь с этим, хотя может быть трудно однозначно и последовательно идентифицировать дисплей.

3 голосов
/ 29 апреля 2009

Я нашел решение вашей проблемы, написав небольшую функцию, которая проверяет, находится ли poitn на подключенном экране. Основная идея пришла от http://msdn.microsoft.com/en-us/library/system.windows.forms.screen(VS.80).aspx но некоторые модификации были необходимы.

public static bool ThisPointIsOnOneOfTheConnectedScreens(Point thePoint)
    {
        bool FoundAScreenThatContainsThePoint = false;

        for(int i = 0; i < Screen.AllScreens.Length; i++)
        {
            if(Screen.AllScreens[i].Bounds.Contains(thePoint))
                FoundAScreenThatContainsThePoint = true;
        }
        return FoundAScreenThatContainsThePoint;
    }
1 голос
/ 13 июня 2015

Существует несколько проблем с вышеуказанным решением.

На нескольких экранах, а также, если экран восстановления меньше.

Следует использовать Содержит (...), а не IntersectsWith , так как в противном случае управляющая часть формы может находиться за пределами области экрана.

Я предложу что-нибудь в этом духе

bool TestBounds(Rectangle R) {
    if (MdiParent == null) return Screen.AllScreens.Any(ø => ø.Bounds.Contains(R)); // If we don't have an MdiParent, make sure we are entirely on a screen
    var c = MdiParent.Controls.OfType<MdiClient>().FirstOrDefault(); // If we do, make sure we are visible within the MdiClient area
    return (c != null && c.ClientRectangle.Contains(R));
}

и используется вот так. (Обратите внимание, что я позволяю Windows обрабатывать это, если сохраненные значения не работают)

bool BoundsOK=TestBounds(myBounds);
if (!BoundsOK) {
    myBounds.Location = new Point(8,8); // then try (8,8) , to at least keep the size, if other monitor not currently present, or current is lesser
    BoundsOK = TestBounds(myBounds);
}
if (BoundsOK) { //Only change if acceptable, otherwise let Windows handle it
    StartPosition = FormStartPosition.Manual;
    Bounds = myBounds;
    WindowState = Enum.IsDefined(typeof(FormWindowState), myWindowState) ? (FormWindowState)myWindowState : FormWindowState.Normal;
}
0 голосов
/ 30 января 2009

Попробуйте порождать вашу главную форму в сохраненном месте в восстановленном (не максимизированном) состоянии, ТО затем максимизируйте ее, если последнее состояние было максимизировано.

Как сказал Стю, будьте осторожны с удаленными мониторами в этом случае. Так как сохраненное местоположение может содержать закадровые координаты (даже отрицательные), вы можете фактически получить и невидимое (за пределами экрана) окно. Я думаю, что проверка границ рабочего стола перед загрузкой предыдущего состояния должна предотвратить это.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...