我有一个可能在其他面板中或可能不在其他面板中的面板。我正在尝试找出该面板的可见边界。我认为GetWindowRect
或GetClientRect
可以做到这一点,但没有快乐。
为了测试这个,我创建了一个带有面板和多行文本框的表单。面板比表格大(即它在表格底部下方伸展)
因此,如果我的表格是300乘300,而小组位于10,10,则为100乘500 我想要的东西会告诉我可见区域是100,290(假设面板的0,0相对起点将是10,10。)
这种方法存在吗?
我尝试了一些不同的方法,例如获取我感兴趣的面板的父句柄并测试它,但我不能总是确定直接父级将决定可见性。
以下是我为测试GetWindowRect
或GetClientRect
编写的测试应用程序的代码:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
using System.Runtime.InteropServices;
namespace clientRectTest
{
public partial class Form1 : Form
{
[StructLayout(LayoutKind.Sequential)]
public struct RECT
{
public Int32 left;
public Int32 top;
public Int32 right;
public Int32 bottom;
public static RECT FromRectangle(Rectangle rectangle)
{
RECT win32rect = new RECT();
win32rect.top = rectangle.Top;
win32rect.bottom = rectangle.Bottom;
win32rect.left = rectangle.Left;
win32rect.right = rectangle.Right;
return win32rect;
}
}
[DllImport("user32.dll")]
[return: MarshalAs(UnmanagedType.Bool)]
public static extern bool GetClientRect(IntPtr hWnd, out RECT lpRect);
[DllImport("user32.dll")]
[return: MarshalAs(UnmanagedType.Bool)]
public static extern bool GetWindowRect(IntPtr hWnd, out RECT lpRect);
[DllImport("user32.dll")]
[return: MarshalAs(UnmanagedType.Bool)]
public static extern bool SetWindowPos(IntPtr hWnd, IntPtr hWndInsertAfter, int X, int Y, int cx, int cy, uint uFlags);
[DllImport("user32.dll")]
public static extern IntPtr GetWindowDC(IntPtr hWnd);
public Form1()
{
InitializeComponent();
this.AutoScrollMinSize = new Size(250, 500);
}
protected override void OnMouseClick(MouseEventArgs e)
{
NewLine("Click Location: " + e.Location.ToString());
base.OnMouseClick(e);
}
protected override void OnResize(EventArgs e)
{
this.textBox1.Text = "Panel Size" + this.panel1.Size.ToString();
NewLine( "Form Size" + this.Size.ToString());
// NewLine("Panel PARENT Client Rectangle: " + getPanelClientRectangle(this.panel1.Parent.Handle));
NewLine("Panel Client Rectangle: " + getPanelClientRectangle(this.panel1.Handle));
NewLine("Panel Window Rectangle: " + getPanelWindowRectangle(this.panel1.Handle));
NewLine("Panel Window Bounts: " + this.panel1.Bounds.ToString());
NewLine("Panel DC Client Rectangle: " + getPanelDCClientRectangle(this.panel1.Handle));
NewLine("Panel DC Window Rectangle: " + getPanelDCWindowRectangle(this.panel1.Handle));
NewLine("Panel Location: " + this.panel1.Location.ToString());
base.OnResize(e);
}
private string getPanelDCClientRectangle(IntPtr handle)
{
string cr = string.Empty;
RECT r1 = new RECT();
IntPtr dc = GetWindowDC(handle);
GetClientRect(dc, out r1);
cr = r1.left.ToString() + ", " + r1.top.ToString() + ", " + r1.right.ToString()
+ ", " + r1.bottom.ToString();
Point thisLocation = this.Location;
return cr;
}
private string getPanelDCWindowRectangle(IntPtr handle)
{
string cr = string.Empty;
RECT r1 = new RECT();
IntPtr dc = GetWindowDC(handle);
GetWindowRect(dc, out r1);
cr = r1.left.ToString() + ", " + r1.top.ToString() + ", " + r1.right.ToString()
+ ", " + r1.bottom.ToString();
Point thisLocation = this.Location;
return cr;
}
private string getPanelWindowRectangle(IntPtr handle)
{
string cr = string.Empty;
RECT r1 = new RECT();
GetWindowRect(handle, out r1);
cr = r1.left.ToString() + ", " + r1.top.ToString() + ", " + r1.right.ToString()
+ ", " + r1.bottom.ToString();
Point thisLocation = this.Location;
return cr;
}
private string getPanelClientRectangle(IntPtr handle)
{
string cr = string.Empty;
RECT r1 = new RECT();
GetClientRect(handle, out r1);
cr = r1.left.ToString() + ", " + r1.top.ToString() + ", " + r1.right.ToString()
+ ", " + r1.bottom.ToString();
Point thisLocation = this.Location;
return cr;
}
private void NewLine(string p)
{
this.textBox1.Text += Environment.NewLine + p;
}
}
}
编辑:发现更多信息:
我认为行this.AutoScrollMinSize = new Size(250, 500);
搞砸了我的结果。这似乎使得面板500即使它没有显示500.将重做我的一些测试用例。我不会用这条线来引起问题。
答案 0 :(得分:2)
您应该能够使用面板的DisplayRectangle属性来获取当前显示的矩形。此属性在Control中实现,但由ScrollableControl(System.Windows.Forms.Panel的父级)覆盖
答案 1 :(得分:0)
这就是我最终想出的:
WinSDK.RECT parentRect = new WinSDK.RECT();
WinSDK.RECT intersectRect = new WinSDK.RECT();
Rectangle parentRectangle = new Rectangle();
Rectangle intersectRectangle = new Rectangle();
// Get the current Handle.
IntPtr currentHandle = this.Handle;
// Get next Parent Handle.
IntPtr parentHandle = WinSDK.GetParent(this.Handle);
// Get the Rect for the current Window.
WinSDK.GetWindowRect(this.Handle, out intersectRect);
// Load Current Window Rect into a System.Drawing.Rectangle
intersectRectangle = new Rectangle(intersectRect.left, intersectRect.top, intersectRect.right - intersectRect.left, intersectRect.bottom - intersectRect.top );
// Itterate through all parent windows and get the overlap of the visible areas to find out what's actually visible.
while (parentHandle != IntPtr.Zero)
{
// Get the Rect for the Parent Window.
WinSDK.GetWindowRect(parentHandle, out parentRect);
parentRectangle = new Rectangle(parentRect.left, parentRect.top, parentRect.right - parentRect.left, parentRect.bottom - parentRect.top);
// Get the intersect between the current and parent window.
intersectRectangle.Intersect(parentRectangle);
// Set up for next loop.
currentHandle = parentHandle;
parentHandle = WinSDK.GetParent(currentHandle);
}
return intersectRectangle;