标题:winform 支持背景透明的textbox 出现内存泄漏,大家帮忙看看都哪些地方需要 ...
只看楼主
qingzang0
Rank: 1
等 级:新手上路
帖 子:2
专家分:0
注 册:2009-9-18
结帖率:0
已结贴  问题点数:20 回复次数:3 
winform 支持背景透明的textbox 出现内存泄漏,大家帮忙看看都哪些地方需要修改?
代码是codeproject上的,功能很强大, 但内存泄漏,大概每秒30-100k不等,找到了几个地方,但是似乎还有很多地方需要修改
请大家帮忙看看

AlphaBlendTextBox .cs代码如下

using System;
using System.Collections;
using
using System.Drawing;
using System.Data;
using System.Windows.Forms;

using System.Drawing.Imaging;

namespace ZBobb
{
/// <summary>
/// AlphaBlendTextBox: A .Net textbox that can be translucent to the background.
/// (C) 2003 Bob Bradley / ZBobb@
/// </summary>
///


public class AlphaBlendTextBox : System.Windows.Forms.TextBox
{
#region private variables

private uPictureBox myPictureBox;
private  bool  myUpToDate = false;
private  bool  myCaretUpToDate = false;
private  Bitmap myBitmap;
private  Bitmap myAlphaBitmap;

private int myFontHeight = 10;

private System.Windows.Forms.Timer myTimer1;

private bool myCaretState = true;

private bool myPaintedFirstTime = false;

private Color myBackColor = Color.White;
private int myBackAlpha = 10;

/// <summary>
/// Required designer variable.
/// </summary>
private components = null;

#endregion // end private variables


#region public methods and overrides

public AlphaBlendTextBox()
{
// This call is required by the Windows.Forms Form Designer.
InitializeComponent();
// TODO: Add any initialization after the InitializeComponent call

this.BackColor = myBackColor;

this.SetStyle(ControlStyles.UserPaint,false);
this.SetStyle(ControlStyles.AllPaintingInWmPaint,true);
this.SetStyle(ControlStyles.DoubleBuffer,true);


myPictureBox = new uPictureBox();
this.Controls.Add(myPictureBox);
myPictureBox.Dock = DockStyle.Fill;
}


protected override void OnResize(EventArgs e)
{

base.OnResize (e);
this.myBitmap = new Bitmap(this.ClientRectangle.Width,this.ClientRectangle.Height);//(this.Width,this.Height);
this.myAlphaBitmap = new Bitmap(this.ClientRectangle.Width,this.ClientRectangle.Height);//(this.Width,this.Height);
myUpToDate = false;
this.Invalidate();
}


//Some of these should be moved to the WndProc later

protected override void OnKeyDown(KeyEventArgs e)
{
base.OnKeyDown (e);
myUpToDate = false;
this.Invalidate();
}

protected override void OnKeyUp(KeyEventArgs e)
{
base.OnKeyUp (e);
myUpToDate = false;
this.Invalidate();

}

protected override void OnKeyPress(KeyPressEventArgs e)
{
base.OnKeyPress (e);
myUpToDate = false;
this.Invalidate();
}

protected override void OnMouseUp(MouseEventArgs e)
{
base.OnMouseUp (e);
this.Invalidate();
}

protected override void OnGiveFeedback(GiveFeedbackEventArgs gfbevent)
{
base.OnGiveFeedback (gfbevent);
myUpToDate = false;
this.Invalidate();
}


protected override void OnMouseLeave(EventArgs e)
{
//found this code to find the current cursor location
//at http://www.

Point ptCursor = Cursor.Position;

Form f = this.FindForm();
ptCursor = f.PointToClient(ptCursor);
if( !this.Bounds.Contains(ptCursor) )
base.OnMouseLeave (e);
}


protected override void OnChangeUICues(UICuesEventArgs e)
{
base.OnChangeUICues (e);
myUpToDate = false;
this.Invalidate();
}


//--
protected override void OnGotFocus(EventArgs e)
{
base.OnGotFocus (e);
myCaretUpToDate = false;
myUpToDate = false;
this.Invalidate();


myTimer1 = new System.Windows.Forms.Timer();
myTimer1.Interval = (int) win32.GetCaretBlinkTime(); //  usually around 500;

myTimer1.Tick +=new EventHandler(myTimer1_Tick);
myTimer1.Enabled = true;

}

protected override void OnLostFocus(EventArgs e)
{
base.OnLostFocus (e);
myCaretUpToDate = false;
myUpToDate = false;
this.Invalidate();

myTimer1.Dispose();
}

//--

protected override void OnFontChanged(EventArgs e)
{
if (this.myPaintedFirstTime)
this.SetStyle(ControlStyles.UserPaint,false);

base.OnFontChanged (e);

if (this.myPaintedFirstTime)
this.SetStyle(ControlStyles.UserPaint,true);


myFontHeight = GetFontHeight();


myUpToDate = false;
this.Invalidate();
}

protected override void OnTextChanged(EventArgs e)
{
base.OnTextChanged (e);
myUpToDate = false;
this.Invalidate();
}


protected override void WndProc(ref Message m)
{

base.WndProc (ref m);

// need to rewrite as a big switch

if (m.Msg == win32.WM_PAINT)
{
myPaintedFirstTime = true;

if (!myUpToDate || !myCaretUpToDate)
GetBitmaps();
myUpToDate = true;
myCaretUpToDate = true;

if (myPictureBox.Image != null) myPictureBox.Image.Dispose();
myPictureBox.Image = (Image)myAlphaBitmap.Clone();

}

else if (m.Msg ==  win32.WM_HSCROLL || m.Msg == win32.WM_VSCROLL)
{
myUpToDate = false;
this.Invalidate();
}

else if (m.Msg == win32.WM_LBUTTONDOWN
|| m.Msg == win32.WM_RBUTTONDOWN
|| m.Msg == win32.WM_LBUTTONDBLCLK
//  || m.Msg == win32.WM_MOUSELEAVE  ///****
)
{
myUpToDate = false;
this.Invalidate();
}

else if (m.Msg == win32.WM_MOUSEMOVE )
{
if (m.WParam.ToInt32() != 0)  //shift key or other buttons
{
myUpToDate = false;
this.Invalidate();
}
}



//System.Diagnostics.Debug.WriteLine("Pro: " + m.Msg.ToString("X"));

}
/// <summary>
/// Clean up any resources being used.
/// </summary>
protected override void Dispose( bool disposing )
{
if( disposing )
{
//this.BackColor = Color.Pink;
if(components != null)
{
components.Dispose();
}
}
base.Dispose( disposing );
}

#endregion //end public method and overrides


#region public property overrides

public new BorderStyle BorderStyle
{
get {return base.BorderStyle;}
set
{
if (this.myPaintedFirstTime)
this.SetStyle(ControlStyles.UserPaint,false);

base.BorderStyle = value;

if (this.myPaintedFirstTime)
this.SetStyle(ControlStyles.UserPaint,true);

this.myBitmap = null;
this.myAlphaBitmap = null;
myUpToDate = false;
this.Invalidate();
}
}

public  new Color BackColor
{
get
{
return Color.FromArgb(base.BackColor.R, base.BackColor.G, base.BackColor.B);
}
set
{
myBackColor = value;
base.BackColor = value;
myUpToDate = false;
}
}
public override bool Multiline
{
get{return base.Multiline;}
set
{
if (this.myPaintedFirstTime)
this.SetStyle(ControlStyles.UserPaint,false);

base.Multiline = value;

if (this.myPaintedFirstTime)
this.SetStyle(ControlStyles.UserPaint,true);

this.myBitmap = null;
this.myAlphaBitmap = null;
myUpToDate = false;
this.Invalidate();
}
}


#endregion    //end public property overrides

#region private functions and classes

private int GetFontHeight()
{
Graphics g = this.CreateGraphics();
SizeF sf_font = g.MeasureString("X",this.Font);
g.Dispose();
return  (int) sf_font.Height;
}


private void GetBitmaps()
{

if (myBitmap == null
|| myAlphaBitmap == null
|| myBitmap.Width != Width
|| myBitmap.Height != Height
|| myAlphaBitmap.Width != Width
|| myAlphaBitmap.Height != Height)
{
myBitmap = null;
myAlphaBitmap = null;
}


if (myBitmap == null)
{
myBitmap = new Bitmap(this.ClientRectangle.Width,this.ClientRectangle.Height);//(Width,Height);
myUpToDate = false;
}


if (!myUpToDate)
{
//Capture the TextBox control window

this.SetStyle(ControlStyles.UserPaint,false);

win32.CaptureWindow(this,ref myBitmap);

this.SetStyle(ControlStyles.UserPaint,true);
this.SetStyle(ControlStyles.SupportsTransparentBackColor,true);
this.BackColor = Color.FromArgb(myBackAlpha,myBackColor);

}
//--



Rectangle r2 = new Rectangle(0,0,this.ClientRectangle.Width,this.ClientRectangle.Height);
ImageAttributes tempImageAttr = new ImageAttributes();


//Found the color map code in the MS Help

ColorMap[] tempColorMap = new ColorMap[1];
tempColorMap[0] = new ColorMap();
tempColorMap[0].OldColor = Color.FromArgb(255,myBackColor);
tempColorMap[0].NewColor = Color.FromArgb(myBackAlpha,myBackColor);

tempImageAttr.SetRemapTable(tempColorMap);

if (myAlphaBitmap != null)
myAlphaBitmap.Dispose();


myAlphaBitmap = new Bitmap(this.ClientRectangle.Width,this.ClientRectangle.Height);//(Width,Height);

Graphics tempGraphics1 = Graphics.FromImage(myAlphaBitmap);

tempGraphics1.DrawImage(myBitmap,r2,0,0,this.ClientRectangle.Width,this.ClientRectangle.Height,GraphicsUnit.Pixel,tempImageAttr);

tempGraphics1.Dispose();

//----

if (this.Focused && (this.SelectionLength == 0))
{
Graphics tempGraphics2 = Graphics.FromImage(myAlphaBitmap);
if (myCaretState)
{
//Draw the caret
Point caret = this.findCaret();
Pen p = new Pen(this.ForeColor,3);
tempGraphics2.DrawLine(p,caret.X,caret.Y + 0,caret.X,caret.Y + myFontHeight);
tempGraphics2.Dispose();
}

}


}



    * 对我有用[0]
    * 丢个板砖[0]
    * 引用
    * 举报
    * 管理
    * TOP

回复次数:5

    * qingzang1用户头像
    * qingzang1
    * (qingzang1)
    *

    * 等 级:
    *

 
#1楼 得分:0回复于:2010-01-07 23:03:23
//我继续贴

private Point findCaret()
{

Point pointCaret = new Point(0);
int i_char_loc = this.SelectionStart;
IntPtr pi_char_loc = new IntPtr(i_char_loc);

int i_point = win32.SendMessage(this.Handle,win32.EM_POSFROMCHAR,pi_char_loc,IntPtr.Zero);
pointCaret = new Point(i_point);

if (i_char_loc == 0)
{
pointCaret = new Point(0);
}
else if (i_char_loc >= this.Text.Length)
{
pi_char_loc = new IntPtr(i_char_loc - 1);
i_point = win32.SendMessage(this.Handle,win32.EM_POSFROMCHAR,pi_char_loc,IntPtr.Zero);
pointCaret = new Point(i_point);

Graphics g = this.CreateGraphics();
String t1 = this.Text.Substring(this.Text.Length-1,1) + "X";
SizeF sizet1 = g.MeasureString(t1,this.Font);
SizeF sizex  = g.MeasureString("X",this.Font);
g.Dispose();
int xoffset = (int)(sizet1.Width - sizex.Width);
pointCaret.X = pointCaret.X + xoffset;

if (i_char_loc == this.Text.Length)
{
String slast = this.Text.Substring(Text.Length-1,1);
if (slast == "\n")
{
pointCaret.X = 1;
pointCaret.Y = pointCaret.Y + myFontHeight;
}
}

}


return pointCaret;
}


private void myTimer1_Tick(object sender, EventArgs e)
{
//Timer used to turn caret on and off for focused control

myCaretState = !myCaretState;
myCaretUpToDate = false;
this.Invalidate();
}


private class uPictureBox : PictureBox
{
public uPictureBox()
{
this.SetStyle(ControlStyles.Selectable,false);
this.SetStyle(ControlStyles.UserPaint,true);
this.SetStyle(ControlStyles.AllPaintingInWmPaint,true);
this.SetStyle(ControlStyles.DoubleBuffer,true);

this.Cursor = null;
this.Enabled = true;
this.SizeMode = PictureBoxSizeMode.Normal;

}


//uPictureBox
protected override void WndProc(ref Message m)
{
if (m.Msg == win32.WM_LBUTTONDOWN
|| m.Msg == win32.WM_RBUTTONDOWN
|| m.Msg == win32.WM_LBUTTONDBLCLK
|| m.Msg == win32.WM_MOUSELEAVE
|| m.Msg == win32.WM_MOUSEMOVE )
{
//Send the above messages back to the parent control
win32.PostMessage(this.Parent.Handle,(uint) m.Msg,m.WParam,m.LParam);
}

else if (m.Msg == win32.WM_LBUTTONUP)
{
//??  for selects and such
this.Parent.Invalidate();
}


base.WndProc (ref m);
}


}  // End uPictureBox Class


#endregion  // end private functions and classes


#region Component Designer generated code
/// <summary>
/// Required method for Designer support - do not modify
/// the contents of this method with the code editor.
/// </summary>
private void InitializeComponent()
{
components = new ();
}
#endregion


#region New Public Properties

[
Category("Appearance"),
Description("The alpha value used to blend the control's background. Valid values are 0 through 255."),
Browsable(true),
DesignerSerializationVisibility(DesignerSerializationVisibility.Visible)

]
public int BackAlpha
{
get { return myBackAlpha; }
set
{
int v = value;
if (v > 255)
v = 255;
myBackAlpha = v;
myUpToDate = false;
Invalidate();
}
}

#endregion


}  // End AlphaTextBox Class


}  // End namespace ZBobb


//----


 


win32.cs代码如下
using System;
using System.Runtime.InteropServices;

using System.Drawing;


namespace ZBobb
{
/// <summary>
/// Win32 support code.
/// (C) 2003 Bob Bradley / ZBobb@
/// </summary>
public class win32
{

public const int  WM_MOUSEMOVE      =              0x0200;
public const int  WM_LBUTTONDOWN    =            0x0201;
public const int  WM_LBUTTONUP      =            0x0202;
public const int  WM_RBUTTONDOWN    =            0x0204;
public const int  WM_LBUTTONDBLCLK  =            0x0203;

public const int  WM_MOUSELEAVE      =            0x02A3;



public const int WM_PAINT    =                  0x000F;
public const int WM_ERASEBKGND  =                0x0014;

public const int WM_PRINT        =              0x0317;

//const int EN_HSCROLL      =  0x0601;
//const int EN_VSCROLL      =  0x0602;

public const int WM_HSCROLL      =              0x0114;
public const int WM_VSCROLL      =              0x0115;


public const int EM_GETSEL              = 0x00B0;
public const int EM_LINEINDEX          = 0x00BB;
public const int EM_LINEFROMCHAR        = 0x00C9;

public const int EM_POSFROMCHAR        = 0x00D6;


[DllImport("USER32.DLL", EntryPoint= "PostMessage")]
public static extern bool PostMessage(IntPtr hwnd, uint msg,
IntPtr wParam, IntPtr lParam);

/*
BOOL PostMessage(          HWND hWnd,
UINT Msg,
WPARAM wParam,
LPARAM lParam
);
*/

// Put this declaration in your class  //IntPtr
[DllImport("USER32.DLL", EntryPoint= "SendMessage")]
public static extern int SendMessage(IntPtr hwnd, int msg, IntPtr wParam,
IntPtr lParam);



[DllImport("USER32.DLL", EntryPoint= "GetCaretBlinkTime")]
public static extern uint GetCaretBlinkTime();


const int WM_PRINTCLIENT = 0x0318;

const long PRF_CHECKVISIBLE=0x00000001L;
const long PRF_NONCLIENT = 0x00000002L;
const long PRF_CLIENT = 0x00000004L;
const long PRF_ERASEBKGND = 0x00000008L;
const long PRF_CHILDREN = 0x00000010L;
const long PRF_OWNED = 0x00000020L;

/*  Will clean this up later doing something like this
enum  CaptureOptions : long
{
PRF_CHECKVISIBLE= 0x00000001L,
PRF_NONCLIENT = 0x00000002L,
PRF_CLIENT = 0x00000004L,
PRF_ERASEBKGND = 0x00000008L,
PRF_CHILDREN = 0x00000010L,
PRF_OWNED = 0x00000020L
}
*/


public static bool CaptureWindow(System.Windows.Forms.Control control,
ref System.Drawing.Bitmap bitmap)
{
//This function captures the contents of a window or control

Graphics g2 = Graphics.FromImage(bitmap);

//PRF_CHILDREN // PRF_NONCLIENT
int meint = (int)(PRF_CLIENT | PRF_ERASEBKGND); //| PRF_OWNED ); //  );
System.IntPtr meptr = new System.IntPtr(meint);

System.IntPtr hdc = g2.GetHdc();
win32.SendMessage(control.Handle,win32.WM_PRINT,hdc,meptr);

g2.ReleaseHdc(hdc);
g2.Dispose();

return true;

}


}
}
搜索更多相关主题的帖子: 内存 textbox 泄漏 winform 
2010-01-07 23:22
qingzang0
Rank: 1
等 级:新手上路
帖 子:2
专家分:0
注 册:2009-9-18
得分:0 
自己顶一下
2010-01-08 10:44
zhoufeng1988
Rank: 15Rank: 15Rank: 15Rank: 15Rank: 15
来 自:北京
等 级:贵宾
威 望:27
帖 子:1432
专家分:6329
注 册:2009-5-31
得分:20 
C#一般是不会出现内存泄露的。因为所有的垃圾资源都由GC释放。
我没仔细看的代码。
C++代码里很容易出现内存泄露,我想可能是你调用非托管代码时出现的问题。
2010-01-08 11:57
amitacai
Rank: 1
等 级:新手上路
帖 子:1
专家分:0
注 册:2015-5-7
得分:0 
回复 3楼 zhoufeng1988
你的代码虽然多只是张贴了一部分.如果你尚未解决.我认为你可以查:那些块模分配内存没有及时释放内存.
2015-05-27 16:25



参与讨论请移步原网站贴子:https://bbs.bccn.net/thread-295245-1-1.html




关于我们 | 广告合作 | 编程中国 | 清除Cookies | TOP | 手机版

编程中国 版权所有,并保留所有权利。
Powered by Discuz, Processed in 0.662164 second(s), 7 queries.
Copyright©2004-2025, BCCN.NET, All Rights Reserved