9

我创建了一个表单可以继承的类,它处理表单位置、大小和状态。而且效果很好。除了一件事:

当您在与主屏幕不同的屏幕上最大化应用程序时,位置和大小(在您最大化之前)会正确存储,但是当它被最大化时(根据其先前的状态),它会在我的主监视器上最大化。然后当我将它恢复到正常状态时,它会转到之前的另一个屏幕。当我再次最大化它时,它当然会在正确的屏幕上最大化。

所以我的问题是......我怎样才能制作一个表格,当它最大化时,记住它是在哪个屏幕上最大化的?当表单再次打开时,我该如何恢复?


问题的完整解决方案

我接受了答案,该答案对如何在屏幕上显示有很好的提示。但这只是我的问题的一部分,所以这是我的解决方案:

负载

  1. 首先存储BoundsWindowState从任何存储中获取。
  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. 如果WindowStateFormWindowState.Normal,则存储Bounds,否则存储RestoreBounds

就是这样!=)

一些示例代码

所以,正如Oliver所建议的,这里有一些代码。它需要充实,但这可以作为任何想要的人的开始:

持久表单处理程序

负责在某处存储和获取数据。

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:在加载方法中,您必须查找先前存储WindowStateWindowBounds和其他值。我们使用 SQL Server,并有一个包含, , (for ), , , , , ,Window列的表。因此,对于每个窗口,对于每个用户和机器,您都会有一行带有, , ,和。此外,我们有一个表,它只有一个外键,一个type列和一个type 列。如果有没有找到的东西,我只保留默认值并返回 false。IdNameMachineNameEnvironment.MachineNameUserIdWindowStateXYHeightWidthWindowStateXYHeightWidthWindowValuesWindowIdKeyStringValueBinary

注意 2:在 save 方法中,当然与在 Load 方法中所做的相反。为当前用户和机器创建行WindowWindowValues如果它们不存在。

持久表单库

这个类使用前一个类并为其他形式形成了一个方便的基类。

// 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 继承。这将自动处理边界和状态。如果应该存储其他任何内容,例如拆分器距离,您将侦听ValuesLoadedandStoringValues事件并在这些事件中使用GetValueandStoreValue方法。

希望这可以帮助某人!如果有,请告诉我。另外,如果您认为有什么可以做得更好或什么的,请提供一些反馈。我想学=)

4

4 回答 4

4

没有内置的方法可以做到这一点 - 你必须自己编写逻辑。原因之一是您必须决定如何处理上次显示窗口的监视器不再可用的情况。例如,这在笔记本电脑和投影仪中很常见。Screen类具有一些有用的功能来帮助解决此问题,尽管可能难以唯一且一致地识别显示。

于 2009-01-30T12:41:37.783 回答
3

我通过编写一个小函数找到了解决您问题的方法,该函数可以测试某个点是否在连接的屏幕上。主要思想来自 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;
    }
于 2009-04-29T00:30:10.323 回答
1

上述解决方案存在一些问题。

在多个屏幕上以及还原屏幕较小时。

It should use Contains(...), rather than IntersectsWith as the control part of the form might otherwise be outside the screen-area.

I will suggest something along these lines

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));
}

and used like this. (Note that I let Windows handle it if the saved values does not work)

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;
}
于 2015-06-13T13:18:54.150 回答
0

尝试以恢复(非最大化)状态在其保存的位置生成主窗体,如果最后一个状态已最大化,则将其最大化。

正如 Stu 所说,在这种情况下要小心移除监视器。由于保存的位置可能包含屏幕外坐标(甚至是负坐标),因此您可能会有效地结束和不可见(实际上是屏幕外)的窗口。我认为在加载以前的状态之前检查桌面边界应该可以防止这种情况。

于 2009-01-30T13:07:17.590 回答