现在的位置: 首页 > 综合 > 正文

如何设置Winform控件的ClientRectangle

2012年10月11日 ⁄ 综合 ⁄ 共 3501字 ⁄ 字号 评论关闭
       最近学习制作WinForm控件,自己动手写控件的时候才发现System.Windows.Forms.Control 竟然没有提供默认的border绘制。记得以前用API做控件的时候,只需要设置空间窗口的WS_BORDER 风格就可以。遍寻无方,只有自己绘制了,这里有出现一个,如果border在客户区,那么在OnPaint方法里不得不每次都要考虑Border所占用的区域,而且,如果从这个类派生的话,将无法获得准确的客户区。
      现在要解决的问题就是如何重新设置客户区的矩形区域的尺寸,查看了一下Control类的ClientRectangle属性:
public Rectangle ClientRectangle { get; }是个只读属性,看来是不能通过这个属性达到目的了。再查找Control类的文档,也没有这方面的说明,没有办法,只能用API搞定了。可以通过计算非客户区尺寸来设置客户区尺寸,Border在非客户绘制。下面就是主要的代码,就是通过重载WndProc方法,捕捉WM_NCCALCSIZE消息,实现自己的逻辑。
     

protected override void WndProc(ref Message m)
{
switch (m.Msg)
{
case (int)WinAPI_WM.WM_NCCALCSIZE:
if (m.WParam.ToInt32() == 0)
{
WinAPI_RECT rc 
= (WinAPI_RECT)m.GetLParam(typeof(WinAPI_RECT));
rc.Left 
+= 1;
rc.Top 
+= 1
rc.Right 
-= 1
rc.Bottom 
-= 1;
Marshal.StructureToPtr(rc, m.LParam, 
true);
m.Result 
= IntPtr.Zero;
}

else
{
WinAPI_NCCALCSIZE_PARAMS csp;
csp 
= (WinAPI_NCCALCSIZE_PARAMS)m.GetLParam(typeof(WinAPI_NCCALCSIZE_PARAMS));
csp.rgrc0.Top 
+= 1
csp.rgrc0.Bottom 
-= 1;
csp.rgrc0.Left 
+= 1
csp.rgrc0.Right 
-= 1;

Marshal.StructureToPtr(csp, m.LParam, 
true);
//Return zero to preserve client rectangle
m.Result = IntPtr.Zero;
}

break;
case (int)WinAPI_WM.WM_NCPAINT:
{
m.WParam 
= NCPaint(m.WParam);
break;
}

}


base.WndProc(ref m);
}


public IntPtr NCPaint(IntPtr region)
{
IntPtr hDC 
= GetWindowDC(this.Handle);
if (hDC != IntPtr.Zero)
{
Graphics grTemp 
= Graphics.FromHdc(hDC);

int ScrollBarWidth = SystemInformation.VerticalScrollBarWidth;
int ScrollBarHeight = SystemInformation.HorizontalScrollBarHeight;

WINDOWINFO wi 
= new WINDOWINFO();
wi.cbSize 
= (uint)Marshal.SizeOf(wi);

//得到当前控件的窗口信息
GetWindowInfo(Handle, ref wi);

wi.rcClient.Right
--;
wi.rcClient.Bottom
--;


//获得当前控件的区域
Region UpdateRegion = new Region(new Rectangle(wi.rcWindow.Top,wi.rcWindow.Left,wi.rcWindow.Right-wi.rcWindow.Left,wi.rcWindow.Bottom-wi.rcWindow.Top));

//获得客户区以外的区域
UpdateRegion.Exclude(new Rectangle(wi.rcClient.Top, wi.rcClient.Left, wi.rcClient.Right - wi.rcClient.Left, wi.rcClient.Bottom - wi.rcClient.Top));

if (IsHScrollVisible && IsVScrollVisible)
{
UpdateRegion.Exclude(Rectangle.FromLTRB
(wi.rcClient.Right 
+ 1, wi.rcClient.Bottom + 1,
wi.rcWindow.Right, wi.rcWindow.Bottom));
}


//得到当前区域的句柄
IntPtr hRgn = UpdateRegion.GetHrgn(grTemp);

//For Painting we need to zero offset the Rectangles.
Rectangle WindowRect = new Rectangle(wi.rcWindow.Top, wi.rcWindow.Left, wi.rcWindow.Right - wi.rcWindow.Left, wi.rcWindow.Bottom - wi.rcWindow.Top);

Point offset 
= Point.Empty - (Size)WindowRect.Location;

WindowRect.Offset(offset);

Rectangle ClientRect 
= WindowRect;

ClientRect.Inflate(
-1-1);

//Fill the BorderArea
Region PaintRegion = new Region(WindowRect);
PaintRegion.Exclude(ClientRect);
grTemp.FillRegion(SystemBrushes.Control, PaintRegion);

//Fill the Area between the scrollbars
if (IsHScrollVisible && IsVScrollVisible)
{
Rectangle ScrollRect 
= new Rectangle(ClientRect.Right - ScrollBarWidth,
ClientRect.Bottom 
- ScrollBarHeight, ScrollBarWidth + 2, ScrollBarHeight + 2);
ScrollRect.Offset(
-1-1);
grTemp.FillRectangle(SystemBrushes.Control, ScrollRect);
}


//Adjust ClientRect for Drawing Border.
ClientRect.Inflate(22);
ClientRect.Width
--;
ClientRect.Height
--;

//Draw Outer Raised Border
ControlPaint.DrawBorder3D(grTemp, WindowRect, Border3DStyle.Raised,
Border3DSide.Bottom 
| Border3DSide.Left | Border3DSide.Right | Border3DSide.Top);

//Draw Inner Sunken Border
ControlPaint.DrawBorder3D(grTemp, ClientRect, Border3DStyle.Sunken,
Border3DSide.Bottom 
| Border3DSide.Left | Border3DSide.Right | Border3DSide.Top);

ReleaseDC(Handle, hDC);

grTemp.Dispose();

return hRgn;

}


RefreshScrollBar();
return region;
}

抱歉!评论已关闭.