一个透明的panel (100分)

  • 主题发起人 主题发起人 zzbyy
  • 开始时间 开始时间
Z

zzbyy

Unregistered / Unconfirmed
GUEST, unregistred user!
我有很多可以实现透明或半透明的panel
但是将其覆盖在wincontrol类上无法实现透明效果,是否有在wincontrol类上
也可以透明的panel,或是使我已有的panel能实现这一效果的方法
 
截取WM_ERASEBKGND 消息
void __fastcall TForm1::OnWMEraseBkgnd(TWMEraseBkgnd &Msg)
{
this->Brush->Style=bsClear;
Msg.Result=true;
}
 
好象不能完全透明,在以前已经有讨论过了,可以查询一下就知道了
 
呵呵,为什么不能完全透明?
 
完全可以. 而且实现很简单.
ArJianzeng误会了, Form不能实现完全透明. 但TWinControl类控件(比如Panel, Edit....)
可以. wangxd答对了一半. 为了实现透明, Window Styles 中有两个非常关键的标志
一个是ExStyle里的WS_EX_TRANSPARENT. 另一个是Style中的WS_CLIPCHILDREN.
如果设置了WS_EX_TRANSPARENT标志, 那么该window在刷新时的背景将不会由
系统清成默认底色, 而是复制上底层的图象. 所以要实现透明/半透明, 必须设置该
标志. 设置了该标记, 然后再截取WM_ERASEBKGND消息, 什么都不做直接返回1
就可以实现透明效果.
然而要实现真正的透明/半透明, 还有一个关键标志很多人都会忽略, 那就是WS_CLIPCHILDREN
如果某个window具备这个标志(默认的), 那么当它刷新时, 它将不刷新它的子窗口
所覆盖的区域, 因此, 某个panel要实现透明的话, 必须将这个panel的parent的WS_CLIPCHILDREN
标志去除. 不然的话, 你的panel移动或者panel遮住的地方的改动都不会体现出来.
这也就是为什么Panel能实现透明/半透明而form无法做到的根本原因. 因为panel是
一个子窗口, 我们只要简单用SetWindowLong就可以修改Parent窗口的Style. 不幸
的是, SetWindowLong只能修改本进程内某个window的Style, 无法修改别的进程
Panel和它的Parent肯定是一个进程中的, 所以能够修改而实现透明效果, 但form的
Parent是Desktop, 不属于一个进程, 无法修改它的Style, 因此哪怕你设置了Form
的Style 为WS_EX_TRANSPARENT并且截取了WM_ERASEBKGND消息, 你也
无法得到Desktop上被Form遮住的部分, 只能得到一块白色区域. 究其原因, 就是
Desktop Window的WS_CLIPCHILDREN在作怪.

 
panel实现透明并不是难,问题是被透明的panel遮住的(panel,command...)Twincontrol
都看不见,只有可以看见,应该怎样修改?
 
过去写的透明panel,贴出来供参考。
unit Glass;

interface

uses
Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs,
ExtCtrls;

type
TGlassStyle = (
gsBlackness, gsDstInvert, gsMergeCopy, gsMergePaint, gsNotSrcCopy,
gsNotSrcErase, gsPatCopy, gsPatInvert, gsPatPaint, gsSrcAnd,
gsSrcCopy, gsSrcErase, gsSrcInvert, gsSrcPaint, gsWhiteness);

TGlass = class(TCustomControl)
private
FColor: TColor;
FStyle: TGlassStyle;
FOnPaint: TNotifyEvent;

procedure SetColor(Value: TColor);
procedure SetStyle(Value: TGlassStyle);
procedure CMCtl3DChanged(var Message: TMessage); message CM_CTL3DCHANGED;
procedure WMEraseBkgnd(var Message: TMessage); message WM_ERASEBKGND;
procedure WMWindowPosChanging(var Message: TWMWindowPosChanging); message WM_WINDOWPOSCHANGING;
protected
Buffer: TBitmap;

procedure CreateParams(var Params: TCreateParams); override;
procedure Paint; override;
procedure Resize; override;
public
constructor Create(AOwner: TComponent); override;
destructor Destroy; override;
property Canvas;
published
property Align;
property Anchors;
property AutoSize;
property BiDiMode;
property BorderWidth;
property Color: TColor read FColor write SetColor;
property Ctl3D;
property Enabled;
property Style: TGlassStyle read FStyle write SetStyle default gsSrcAnd;
property Visible;

property OnClick;
property OnDblClick;
property OnEnter;
property OnExit;
property OnMouseDown;
property OnMouseMove;
property OnMouseUp;
property OnResize;
property OnPaint: TNotifyEvent read FOnPaint write FOnPaint;
end;

procedure Register;

implementation

procedure Register;
begin
RegisterComponents('Croco', [TGlass]);
end;

function GlassStyleToInt(gs: TGlassStyle): LongInt;
begin
case gs of
gsBlackness : Result := cmBlackness;
gsDstInvert : Result := cmDstInvert;
gsMergeCopy : Result := cmMergeCopy;
gsMergePaint : Result := cmMergePaint;
gsNotSrcCopy : Result := cmNotSrcCopy;
gsNotSrcErase: Result := cmNotSrcErase;
gsPatCopy : Result := cmPatCopy;
gsPatInvert : Result := cmPatInvert;
gsPatPaint : Result := cmPatPaint;
gsSrcAnd : Result := cmSrcAnd;
gsSrcCopy : Result := cmSrcCopy;
gsSrcErase : Result := cmSrcErase;
gsSrcInvert : Result := cmSrcInvert;
gsSrcPaint : Result := cmSrcPaint;
gsWhiteness : Result := cmWhiteness;
else Assert(True, 'Error parameter in function GlassStyleToInt');
end;
end;

constructor TGlass.Create(AOwner: TComponent);
begin
inherited Create(AOwner);
Buffer := TBitmap.Create;

ControlStyle := [csAcceptsControls, csCaptureMouse, csClickEvents,
csDoubleClicks, csReplicatable];
Width := 100;
Height := 100;
FStyle := gsSrcAnd;
ParentCtl3d := False;
Ctl3D := False;
ParentColor := False;
FColor := clWhite;
end;

destructor TGlass.Destroy;
begin
Buffer.Free;
inherited Destroy;
end;

procedure TGlass.Paint;
var
R: TRect;
rop: LongInt;
begin
R := Rect(0, 0, Width, Height);
Buffer.Width := Width;
Buffer.Height := Height;
Buffer.Canvas.Brush.Style := bsSolid;
Buffer.Canvas.Brush.Color := FColor;
Buffer.Canvas.FillRect(Rect(0, 0, Width, Height));
rop := GlassStyleToInt(FStyle);
StretchBlt(Buffer.Canvas.Handle, 0, 0, Width, Height,
Canvas.Handle, 0, 0, Width, Height, rop);
if Ctl3D then DrawEdge(Buffer.Canvas.Handle, R, BDR_RAISEDINNER, BF_RECT);
Buffer.Canvas.Pen.Mode := pmCopy;
Buffer.Canvas.Pen.Style := psSolid;
Canvas.Draw(0, 0, Buffer);
if Assigned(FOnPaint) then FOnPaint(Self);
end;


procedure TGlass.SetColor(Value: TColor);
begin
if Value <> FColor then
begin
FColor := Value;
RecreateWnd;
end;
end;

procedure TGlass.CreateParams(var Params: TCreateParams);
begin
inherited CreateParams(Params);

Params.ExStyle := Params.ExStyle + WS_EX_TRANSPARENT;
end;

procedure TGlass.WMWindowPosChanging(var Message: TWMWindowPosChanging);
begin
Invalidate;

inherited;
end;

procedure TGlass.WMEraseBkgnd(var Message: TMessage);
begin
Message.Result := 0;
end;

procedure TGlass.Resize;
begin
Invalidate;

inherited;
end;

procedure TGlass.CMCtl3DChanged(var Message: TMessage);
begin
inherited;

RecreateWnd;
end;

procedure TGlass.SetStyle(Value: TGlassStyle);
begin
if Value <> FStyle then
begin
FStyle := Value;
RecreateWnd;
end;
end;

end.
 
接受答案了.
 
后退
顶部