作者:linzhenqun(风)
时间:<chsdate style="BACKGROUND-POSITION: left bottom; BACKGROUND-IMAGE: url(res://ietag.dll/#34/#1001); BACKGROUND-REPEAT: repeat-x" tabindex="0" w:st="on" year="2005" month="8" day="14" islunardate="False" isrocdate="False">2005-8-14</chsdate>
Blog: http://blog.csdn.net/linzhengqun
----------------------------------------------------
上一篇介绍了Hint的简单应用,这一篇将给出一个定制Hint窗口的例子。这个自定义Hint窗口的效果不错,以玻璃为边框,并且有阴影的效果。
不过这之前,我们必须介绍一个如何定制,Hint的父类为THintWindow,在Controls单元中定义。我们看看几个虚拟方法,CreateParams设定窗口的风格,我们要覆盖掉它,使其没有边框。NCPaint画窗口的边框,我们也要覆盖它,因为我们不需要边框吗。Paint比较重要,为画Hint窗口客户区内容,当然要覆盖。不过最重要的当属ActivateHint,它会设定好窗口的大小,并显示它,我们就在这里定制一个类玻璃的窗口效果。下面给出该类的实现:
unit wdHintWnd;
interface
uses
Windows, Classes, Controls, Graphics, Forms, SysUtils, ExtCtrls;
type
TwdHintWnd = class(THintWindow)
private
FWndBmp: TBitmap; //窗口位图
FHintBmp: TBitmap; //提示信息位图
protected
procedure CreateParams(var Params: TCreateParams); override;
procedure Paint; override;
procedure NCPaint(DC: HDC); override;
{画提示的图象}
procedure DrawHintImg(Bmp:TBitmap; AHint: string);
{取得提示窗口对应的桌面区域的图象}
procedure GetDesktopImg(Bmp: TBitmap; R: TRect);
{对桌面区域图象作处理,使其看起来像一块玻璃且带有一点阴影}
procedure EffectHandle(WndBmp, HintBmp: TBitmap);
public
constructor Create(Aowner: TComponent); override;
destructor Destroy; override;
procedure ActivateHint(Rect: TRect; const AHint: string); override;
end;
implementation
{ TwdHintWnd }
procedure TwdHintWnd.ActivateHint(Rect: TRect; const AHint: string);
var
P: TPoint;
begin
//在这里取得一个适当的尺寸显示文字
FHintBmp.Width := Rect.Right - Rect.Left;
FHintBmp.Height := Rect.Bottom - Rect.Top + 4;
DrawHintImg(FHintBmp, AHint);
FWndBmp.Width := Rect.Right - Rect.Left + 23;
FWndBmp.Height := Rect.Bottom - Rect.Top + 27;
Inc(Rect.Right, 23);
Inc(Rect.Bottom, 27);
BoundsRect := Rect;
if Left < Screen.DesktopLeft then
Left := Screen.DesktopLeft;
if Top < Screen.DesktopTop then
Top := Screen.DesktopTop;
if Left + Width > Screen.DesktopWidth then
Left := Screen.DesktopWidth - Width;
if Top + Height > Screen.DesktopHeight then
Top := Screen.DesktopHeight - Height;
GetDesktopImg(FWndBmp, BoundsRect);
EffectHandle(FWndBmp, FHintBmp);
P := ClientToScreen(Point(0, 0));
SetWindowPos(Handle, HWND_TOPMOST, P.X, P.Y, 0, 0,
SWP_SHOWWINDOW or SWP_NOACTIVATE or SWP_NOSIZE);
end;
constructor TwdHintWnd.Create(Aowner: TComponent);
begin
inherited;
FWndBmp := TBitmap.Create;
FWndBmp.PixelFormat := pf24bit;
FHintBmp := TBitmap.Create;
end;
procedure TwdHintWnd.CreateParams(var Params: TCreateParams);
begin
inherited;
//去掉窗口边框
Params.Style := Params.Style and not WS_BORDER;
end;
destructor TwdHintWnd.Destroy;
begin
FWndBmp.Free;
FHintBmp.Free;
inherited;
end;
procedure TwdHintWnd.GetDesktopImg(Bmp: TBitmap; R: TRect);
var
C: TCanvas;
begin
C:= TCanvas.Create;
try
C.Handle := GetDC(0);
Bmp.Canvas.CopyRect(Rect(0, 0, Bmp.Width, Bmp.Height), C, R);
finally
C.Free;
end;
end;
procedure TwdHintWnd.EffectHandle(WndBmp, HintBmp: TBitmap);
var
R: TRect;
i, j: Integer;
P: PByteArray;
Transt, TranstAngle: Integer;
begin
R := Rect(0, 0, WndBmp.Width - 4, WndBmp.Height - 4);
Frame3D(WndBmp.Canvas, R, clMedGray, clBtnShadow, 1);
//作窗口底下的阴影效果
Transt := 60;
for j:= WndBmp.Height - 4 to WndBmp.Height - 1 do
begin
P := WndBmp.ScanLine[j];
TranstAngle := Transt;
for i:= 3 to WndBmp.Width - 1 do
begin
//如果正处于右下角
if i > WndBmp.Width - 5 then
begin
P[3*i] := P[3*i] * TranstAngle div 100;
P[3*i + 1] := P[3*i + 1] * TranstAngle div 100;
P[3*i + 2] := P[3*i + 2] * TranstAngle div 100;
TranstAngle := TranstAngle + 10;
if TranstAngle > 90 then TranstAngle := 90;
end
else begin
P[3*i] := P[3*i] * Transt div 100;
P[3*i + 1] := P[3*i + 1] * Transt div 100;
P[3*i + 2] := P[3*i + 2] * Transt div 100;
end;
end;
Transt := Transt + 10;
end;
//作窗口右边的阴影效果
for j := 3 to WndBmp.Height - 5 do
begin
P := WndBmp.ScanLine[j];
Transt := 60;
for i:= WndBmp.Width - 4 to WndBmp.Width -1 do
begin
P[3*i] := P[3*i] * Transt div 100;
P[3*i + 1] := P[3*i + 1] * Transt div 100;
P[3*i + 2] := P[3*i + 2] * Transt div 100;
Transt := Transt + 10;
end;
end;
WndBmp.Canvas.Draw(10, 10, HintBmp);
end;
procedure TwdHintWnd.NCPaint;
begin
//重载不让画边框
end;
procedure TwdHintWnd.Paint;
begin
Canvas.CopyRect(ClientRect, FWndBmp.Canvas, ClientRect);
end;
procedure TwdHintWnd.DrawHintImg(Bmp: TBitmap; AHint: string);
var
R: TRect;
begin
Bmp.Canvas.Brush.Color := Application.HintColor;
Bmp.Canvas.Pen.Color := Application.HintColor;
Bmp.Canvas.Rectangle(0, 0, Bmp.Width, Bmp.Height);
Bmp.Canvas.Font.Color := Screen.HintFont.Color;
R := Rect(0, 0, Bmp.Width, Bmp.Height);
Inc(R.Left, 2);
Inc(R.Top, 2);
DrawText(Bmp.Canvas.Handle, PChar(AHint), -1, R, DT_LEFT or DT_NOPREFIX or
DT_WORDBREAK or DrawTextBiDiModeFlagsReadingOnly);
end;
initialization
Application.ShowHint := False;
HintWindowClass := TwdHintWnd;
Application.ShowHint := True;
end.
只需将该单元加入你的工程当中,然后运行程序,便可看到效果了,试试看,漂亮吧。
程序中重要部分已经作了注释,这里只说明几个重要的地方,首先是initialization
部分,这里将Application的ShowHint设为False,看一下VCL源码,知道Application将一个HintWindow给消毁了,而HintWindowClass定义如下:
THintWindowClass = class of THintWindow;它是THintWindow的类引用,在Forms单元中它初始化为THintWindow:
HintWindowClass: THintWindowClass = THintWindow;
在这里我们将其替换为TwdHintWnd,最后将ShowHint设为True,Application便用HintWindowClass创建一个Hint窗口,此时创建的便是我们定制的类了,以后的提示窗口就将用我们上面的窗口来显示。
在ActivateHint方法,我们将作效果的处理,原理是取得提示窗口在桌面上的位置对应的位图,然后画到提示窗口上,再将提示信息的位置拷贝到提示窗口中间,这样就有了透明的效果了。其次画出玻璃的边,最后在窗口右边和下边作阴影效果。
关于阴影效果的实现,用到的是图像的Alpha技术,可以到网上找一找,这里就不多说了,只给出图像透明度的公式:
Dst.Red = Src.Red * alpha + (1-alpha) * Dst.Red;
Dst.Green = Src.Green * alpha + (1-alpha) * Dst.Green;
Dst.Blue = Src.Blue * alpha + (1-alpha) * Dst.Blue;
Alpha的值为0到1之间,为1时表示完全不透明,不过我们将用于混合的颜色为黑色,即0,所以上面代码看到的是如下的样子:
P[3*i] := P[3*i] * TranstAngle div 100;
玻璃提示窗口的原理大概如此,当然其透明效果是一个假象,遇到后有动的物体就暴露无疑了。不过作为一个提示窗口,我想已经足够了。
相关推荐
Sherryhint.pas从这儿文件改过来的,很简单,有感兴趣的可以看一下…… Delphi 气泡形式的Hint提示(可以修改字体颜色和边框颜色)
2. **设置Hint**:在创建或加载TMenuItem时,可以为每个菜单项分配Hint信息。例如,`MyMenuItem.Hint := '这是菜单项的提示信息';` 3. **处理OnPopup事件**:TPopupMenu有一个OnPopup事件,在弹出菜单之前触发。...
Delphi中使用Hint体验windows的温柔 根据网友的 意思 。我做了一个测试的demo 现在上传,给爱好delphi的网友 http://blog.csdn.net/SmallHand/archive/2008/12/22/3580217.aspx
delphi hint 在Delphi中每个组件几乎都具有Hint属性,但是他们只是在鼠标移动到该组件的时候才显示Hint内容。幸运的是,Delphi为我们提供了一个THintWindow组件,这样就不需要我们手工封装一个组件了。为了实现我们...
2. **创建Hint机制**: 要实现在下拉列表中显示hint,我们需要监听`OnMouseMove`事件。当鼠标在下拉列表中移动时,此事件会被触发。我们需要在事件处理函数中获取当前鼠标悬停的项,并显示相应的提示信息。 3. **...
### Delphi中的Hint功能详解 #### 一、概述 在Delphi编程中,**Hint**是一种非常实用的功能,主要用于提供用户界面元素上的提示信息。它不仅能够帮助用户更好地理解界面上各个控件的作用,还能提升应用程序的整体...
2. **处理鼠标事件**:在OnMouseEnter事件中,你可以检查当前控件的Hint属性,并根据需要显示自定义的提示窗口。这个窗口可以是TForm、TPopupWindow或者其他任何可以显示文本的组件。在OnMouseLeave事件中,记得隐藏...
3. **处理Hint事件**: Delphi中的TApplicationEvents组件可以监听全局的Hint事件。通过设置OnHint属性,我们可以控制全局的Hint行为。在事件处理函数中,检查当前鼠标位置所在的控件,并决定是否应该显示Hint。 4. ...
"delphi自定义的label_hint单元" 提供了一种方法来扩展标准的TLabel控件,使其具备显示提示信息(Hint)的功能。这在设计用户界面时非常有用,可以为用户提供额外的信息,提高交互体验。下面我们将深入探讨如何实现...
由于cxGrid把Hint给封装了(它的Hint须在列宽不够大时才显示,且显示的内容不能随意),使得Hint无法方便地使用,经在cxGrid源码中分析,弄出这个给cxGrid附加格式化Hint的方法,在Delphi6、ExpressQuantumGrid_v5.8...
一款Delphi文字提示,跟随鼠标光标的文字提示效果,点击“开始”按钮后,激活此效果,把鼠标放在按钮上,即可看到一行滚动的文字提示,在Delphi代码中,变量做如下定义: procedure TimerOn; // 开启光标跟随...
2. 添加`Hint`的自定义显示逻辑: - 在新创建的`TCustomLabelHint`单元中找到`CreateWnd`方法,这里可以初始化一些控件属性,包括`Hint`的初始状态。 - 实现`OnMouseEnter`事件处理程序,当鼠标进入`Label`时,...
"Balloon Hint"是Delphi编程环境中的一种特性,主要用于提供一种可视化的小提示,它会在鼠标悬停在某个控件上时显示一个形如气泡的信息窗口,通常用来展示额外的说明或者帮助信息。在Delphi 2010中,Balloon Hint...
在Delphi编程环境中,"hint.pas" 文件通常与Delphi的源代码提示或提示信息有关。当你在尝试运行一个Delphi程序时,如果遇到"hint.pas(delphi短缺)"这样的描述,这可能意味着你的开发环境中缺少了一个特定的单元...
标题“cell-hint-in-dbgrid.zip_Hint_Selected_dbgrid_delphi cell_delphi”暗示了我们关注的是如何在DBGrid选定单元格中实现提示(Hint)功能。描述进一步确认了这一点,即探讨的是如何在DBGrid组件中选中的单元格...
在Delphi编程环境中,我们经常会遇到需要为控件添加额外信息或提示的需求,这通常可以通过创建自定义的label_hint来实现。"Delphi实现自定义的label_hint"这个主题主要探讨的是如何在Delphi中扩展标准的控件功能,以...
在Delphi编程环境中,自定义控件是提升应用程序功能和用户体验的重要手段。Popup Menu,即弹出式菜单,是用户界面中的一个常见元素,通常用于响应鼠标右键点击或者其他特定事件,显示一系列可选操作。在Delphi中,...
Heir to TComboBox showing Hint for a row in the ListBox partnot completely visible.Author: Igor ShevchenkoDate of creation: 13.04.2002Limitations:The component was checked when working with the Style ...
此文件是一个单元,主要解决TMenuItem的下拉菜单,也就是PopupMenu的Hint属性不管用的问题。 使用方法:下载此单元文件,然后复制到你的项目中,在需要实现功能的Form中引入此单元。然后在Form的onCreate方法中写 ...
在Delphi开发环境中,系统默认的提示信息(Hint)虽然能够提供帮助信息,但其样式相对单一,可能无法满足开发者对于更美观、更直观交互的需求。本文将深入探讨如何创建和应用这种气泡Hint提示,以及它在Delphi编程中...