我正在尝试将没有确定形状的图像(例如帽子)放在不同的图像控件之上。问题是,由于控件具有确定的形状,它会保留默认背景颜色以覆盖留空的空间。图像控件与图像的大小完全相同。我尝试使用 control.BackColor = Color.Transparent; 但它似乎不起作用。还有其他建议吗?
2 回答
您可以Control.Region
用于此目的
GraphicsPath path = new GraphicsPath();
path.AddEllipse(control.ClientRectangle);
control.Region = new Region(path);
试试这个,您可以使用创建任何形状GraphicsPath
并将其设置Region
为例如我创建的椭圆。
编辑
如果你只想设置 BackColor = Color.Transparent。出于某种原因,某些控件不允许这样做。在这种情况下,您可以执行以下操作
public class CustomControl1 : Control
{
public CustomControl1()
{
this.SetStyle(ControlStyles.SupportsTransparentBackColor, true);
}
}
创建控件的后代并设置this.SetStyle(ControlStyles.SupportsTransparentBackColor, true);
应该可以解决问题
如果您的Image Control
(如 a PictureBox
)在运行时没有被用户移动(通过按住鼠标并拖动),您可以使用这种技术,它允许您在彼此的顶部显示图像。图像应具有透明背景:
public class ImageControl : Control {
public ImageControl(){
SetStyle(ControlStyles.Opaque, true);
}
public Image Image {get;set;}
protected override CreateParams CreateParams {
get {
CreateParams cp = base.CreateParams;
cp.ExStyle |= 0x20;
return cp;
}
}
protected override void OnPaint(PaintEventArgs e){
if(Image != null) e.Graphics.DrawImage(Image, Point.Empty);
}
}
您可以使用上面的控件代替PictureBox
. 通过在运行时拖动来移动此控件会导致大量闪烁。因此,如果您愿意,我认为只有 1 个解决方案使用Region
. 在这种方法中,您必须将您的Bitmap
转为 aRegion
并将其分配Region
给您的Control.Region
财产。提供的链接Chris Dunaway
对您执行此操作非常有帮助。但是我不得不说,它的Region
边界并不像你想象的那样平滑。这是这种方法的不足之处。为方便起见,我将在此处发布稍作修改的代码,此代码使用LockBits
的性能将优于原始代码:
public class Util {
//invert will toggle backColor to foreColor (in fact, I mean foreColor here is the Solid Color which makes your image distinct from the background).
public static Region RegionFromBitmap(Bitmap bm, Color backColor, bool invert)
{
Region rgn = new Region();
rgn.MakeEmpty();//This is very important
int argbBack = backColor.ToArgb();
BitmapData data = bm.LockBits(new Rectangle(0, 0, bm.Width, bm.Height), ImageLockMode.ReadOnly, PixelFormat.Format32bppArgb);
int[] bits = new int[bm.Width * bm.Height];
Marshal.Copy(data.Scan0, bits, 0, bits.Length);
//
Rectangle line = Rectangle.Empty;
line.Height = 1;
bool inImage = false;
for (int i = 0; i < bm.Height; i++)
{
for (int j = 0; j < bm.Width; j++)
{
int c = bits[j + i * bm.Width];
if (!inImage)
{
if (invert ? c == argbBack : c != argbBack)
{
inImage = true;
line.X = j;
line.Y = i;
}
}
else if(invert ? c != argbBack : c == argbBack)
{
inImage = false;
line.Width = j - line.X;
rgn.Union(line);
}
}
}
bm.UnlockBits(data);
return rgn;
}
}
//Use the code
//if your Bitmap is a PNG with transparent background, you can get the Region from it like this:
Region rgn = Util.RegionFromBitmap(yourPng, Color.FromArgb(0), false);
//if your Bitmap has a figure with solid color of Black, you can get the Region like this:
Region rgn = Util.RegionFromBitmap(yourPng, Color.Black, true);