使用GDI+绘制的360风格按钮控件
将下面的代码拷贝到一个单元中,创建一个包,加入这个单元后安装.使用的时候设置好背景颜色,边框颜色,图标(png格式)相对路径的文件名称.这个控件可以利用PNG图像的颜色透明特性,背景色默认透明度为50%,可以将按钮后面的内容显示出来.GDIPAPI, GDIPOBJ, GDIPUTIL三个单元可用万一的博客上寻找下载地址.
unit u360StyleButton;
interface
uses
SysUtils, Classes, Controls, StdCtrls,Graphics, Messages, Windows,
GDIPAPI, GDIPOBJ, GDIPUTIL;
type
TBtn360Style = class(TButton)
private
FBkgColor: TColor; //鼠标悬停是的背景颜色
FEdgColor: TColor; //边框颜色
FCanvas: TCanvas;
FMouseEnter: Boolean;
FPngFileName: string;
procedure CNDrawItem(var Message:TWMDrawItem);message CN_DRAWITEM;
procedure SetPngFileName(const Value: string);
procedure SetBkgColor(const Value: TColor);
procedure SetEdgColor(const Value: TColor);
protected
procedure CreateParams(var Params:TCreateParams);override;
procedure CMMouseEnter(var Message: TMessage); message CM_MOUSEENTER;
procedure CMMouseLeave(var Message: TMessage); message CM_MOUSELEAVE;
procedure SetButtonStyle(ADefault:Boolean);override; //必须重新找个函数 否则会按默认样式绘制
public
constructor Create(AOwner:TComponent);override;
destructor Destroy;override;
published
property PngFileName: string read FPngFileName write SetPngFileName;
property BkgColor: TColor read FBkgColor write SetBkgColor;
property EdgColor: TColor read FEdgColor write SetEdgColor;
end;
procedure Register;
implementation
constructor TBtn360Style.Create(AOwner: TComponent);
begin
inherited Create(AOwner);
DoubleBuffered := True;
FCanvas := TCanvas.Create;
FBkgColor := clBlue;
FEdgColor := clSkyBlue;
end;
destructor TBtn360Style.Destroy;
begin
FCanvas.Free;
inherited Destroy;
end;
procedure TBtn360Style.CreateParams(var Params: TCreateParams);
begin
inherited CreateParams(Params);
with Params do Style := Style or BS_OWNERDRAW;
end;
procedure TBtn360Style.SetEdgColor(const Value: TColor);
begin
if FEdgColor <> Value then
begin
FEdgColor := Value;
Invalidate;
end;
end;
procedure TBtn360Style.SetBkgColor(const Value: TColor);
begin
if FBkgColor <> Value then
begin
FBkgColor := Value;
Invalidate;
end;
end;
procedure TBtn360Style.SetButtonStyle(ADefault: Boolean);
begin
if csDesigning in ComponentState then
inherited;
end;
procedure TBtn360Style.SetPngFileName(const Value: string);
begin
if FPngFileName <> Value then
begin
FPngFileName := Value;
Invalidate;
end;
end;
procedure TBtn360Style.CMMouseEnter(var Message: TMessage);
begin
inherited;
FMouseEnter := True;
Invalidate;
end;
procedure TBtn360Style.CMMouseLeave(var Message: TMessage);
begin
inherited;
FMouseEnter := False;
Invalidate;
end;
procedure TBtn360Style.CNDrawItem(var Message: TWMDrawItem);
var
IsDown: Boolean;
ARect: TRect;
DrawItemStruct: TDrawItemStruct;
wh:TSize;
g: TGPGraphics;
pen: TGPPen;
img: TGPImage;
img2: TGPBitmap;
imgAtt: TGPImageAttributes;
i, j:Integer;
const
ColorMatrix: TColorMatrix = (
(1.0, 0.0, 0.0, 0.0, 0.0),
(0.0, 1.0, 0.0, 0.0, 0.0),
(0.0, 0.0, 1.0, 0.0, 0.0),
(0.0, 0.0, 0.0, 1.0, 0.0),
(1.0, 0.0, 0.0, 0.0, 1.0));
begin
DrawItemStruct:=Message.DrawItemStruct^;
FCanvas.Handle := DrawItemStruct.hDC;
g := TGPGraphics.Create(FCanvas.Handle);
pen := TGPPen.Create(GDIPAPI.MakeColor(128, FEdgColor and $FF, (FEdgColor shr 8) and $FF, (FEdgColor shr 16) and $FF));
img := TGPImage.Create(FPngFileName);
img2 := TGPBitmap.Create(Width, Height);
for i := 0 to img2.GetWidth do
for j := 0 to img2.GetHeight do
begin
color := GDIPAPI.MakeColor(128, FBkgColor and $FF, (FBkgColor shr 8) and $FF, (FBkgColor shr 16) and $FF);
img2.SetPixel(i, j, color);
end;
ARect := ClientRect;
with DrawItemStruct do
IsDown := itemState and ODS_SELECTED <> 0;
if FMouseEnter then //鼠标在按钮上 则绘制一个背景及边框
begin
Perform($000B, 0, 0);
g.DrawImage(img2, 0, 0, Width, Height);
g.DrawRectangle(pen, 0, 0, Width - 1, Height - 1);
Perform($000B, 1, 0);
end;
//按钮被按下时的状态绘制
if IsDown then
begin
imgAtt := TGPImageAttributes.Create;
imgAtt.SetColorMatrix(ColorMatrix, ColorMatrixFlagsDefault, ColorAdjustTypeDefault);
g.DrawImage(img, MakeRect(0, 0, img.GetWidth, img.GetHeight),
-10, -10, img.GetWidth + 10, img.GetHeight + 10, UnitPixel, imgAtt);
FreeAndNil(imgAtt);
end
else //绘制一个未按下的按钮
g.DrawImage(img, (Width - img.GetWidth) div 2, 10);
FreeAndNil(img);
FreeAndNil(img2);
FreeAndNil(g);
FreeAndNil(pen);
//绘制Caption文本内容
FCanvas.Font := Self.Font;
ARect:=ClientRect;
wh:=FCanvas.TextExtent(Caption);
FCanvas.Pen.Width := 1;
FCanvas.Brush.Style := bsClear;
if not Enabled then
begin //按钮失效时应多绘一次Caption文本
FCanvas.Font.Color := clBtnHighlight;
FCanvas.TextOut((Width div 2)-(wh.cx div 2), height - wh.cy - 10,Caption);
FCanvas.Font.Color := clBtnShadow;
end
else
FCanvas.TextOut((Width div 2)-(wh.cx div 2), height - wh.cy - 10,Caption);
FCanvas.Handle := 0;
end;
procedure Register;
begin
RegisterComponents('HenreashPackages', [TBtn360Style]);
end;
end.
使用GDI+绘制的360风格按钮控件的更多相关文章
- 使用GDI+绘制的360风格按钮控件(使用CN_DRAWITEM消息重绘,并使用TGPGraphics,TGPPen,TGPImage,TGPBitmap等)good
将下面的代码拷贝到一个单元中,创建一个包,加入这个单元后安装.使用的时候设置好背景颜色,边框颜色,图标(png格式)相对路径的文件名称.这个控件可以利用PNG图像的颜色透明特性,背景色默认透明度为50 ...
- MFC基于对话框风格按钮控件添加图片的方法(大神止步)
菜鸟还在研究这个东西,大神就不要看了.一直都在觉得用VC或VS建立的对话框总是全灰色感觉太单调了,如果可以在上面添加一些漂亮的图片就好了,今天终于实现了.其实挺简单的,下面就分几个步骤讲一下: 第一步 ...
- dskinlite(uieasy mfc界面库)使用记录2:绘制动态元素(按钮控件绘制元素动态控制,改变图片和文字)
效果图:这4个分别是按钮按下后4种状态的效果 第88行是显示默认的按钮文字,没有id,SetWindowText改的就是它了 第87行是左边的图片,id是ico,可以通过程序控制 第89行是蓝色的文字 ...
- dskinlite(uieasy mfc界面库)使用记录3:绘制动态元素(按钮控件通过隐藏方式修改图片显示)
效果图: 分别是:正常,正常鼠标悬停,按下,按下鼠标悬停 XML代码: 75,76行定义了一个image,注意id和index属性 初始化代码: click代码: 147,148,153,154:通过 ...
- Qt编写自定义控件11-设备防区按钮控件
前言 在很多项目应用中,需要根据数据动态生成对象显示在地图上,比如地图标注,同时还需要可拖动对象到指定位置显示,能有多种状态指示,安防领域一般用来表示防区或者设备,可以直接显示防区号,有多种状态颜色指 ...
- Qt编写自定义控件9-导航按钮控件
前言 导航按钮控件,主要用于各种漂亮精美的导航条,我们经常在web中看到导航条都非常精美,都是html+css+js实现的,还自带动画过度效果,Qt提供的qss其实也是无敌的,支持基本上所有的CSS2 ...
- WinRT自定义控件第一 - 转盘按钮控件
之前的文章中,介绍了用WPF做一个转盘按钮控件,后来需要把这个控件移植到WinRT时,遇到了很大的问题,主要原因在于WPF和WinRT还是有很大不同的.这篇文章介绍了这个移植过程,由于2次实现的控件功 ...
- MFC编程入门之二十二(常用控件:按钮控件Button、Radio Button和Check Box)
本节继续讲解常用控件--按钮控件的使用. 按钮控件简介 按钮控件包括命令按钮(Button).单选按钮(Radio Button)和复选框(Check Box)等.命令按钮就是我们前面多次提到的侠义的 ...
- C#自定义Button按钮控件
C#自定义Button按钮控件 在实际项目开发中经常可以遇到.net自带控件并不一定可以满足需要,因此需要自定义开发一些新的控件,自定义控件的办法也有多种,可以自己绘制线条颜色图形等进行重绘,也可以采 ...
随机推荐
- Java 基础 -- BigInteger BigDecimai大数
BigInteger 加减乘除 BigInteger bi1 = new BigInteger("123456789") ; // 声明BigInteger对象 BigIntege ...
- 实战:基于 Spring 的应用配置如何迁移至阿里云应用配置管理 ACM
最近遇到一些开发者朋友,准备将原有的Java Spring的应用配置迁移到 阿里云应用配置管理 ACM 中.迁移过程中,遇到不少有趣的问题.本文将通过一个简单的样例来还原迁移过程中遇到的问题和相关解决 ...
- Vue 事件相关实例方法---on/emit/off/once
一.初始位置 平常项目中写逻辑,避免不了注册/触发各种事件 今天来研究下 Vue 中,我们平常用到的关于 on/emit/off/once 的实现原理 关于事件的方法,是在 Vue 项目下面文件中的 ...
- 【2017中国大学生程序设计竞赛 - 网络选拔赛】Friend-Graph
[链接]http://acm.hdu.edu.cn/showproblem.php?pid=6152 [题意] 有一个队伍,如果队伍里有三个或三个以上的人互相认识 或者队伍里有三个或三个以上的人互相不 ...
- 自己写的一些Delphi常用函数
今天在整理以前写过的代码,发现有些函数还是挺实用的,决定将其贴到Blog上,与众多好友一起分享.{*************************************************** ...
- Python 内置模块:os模块
Python os模块包含普遍的操作系统功能.如果你希望你的程序能够与平台无关的话,这个模块是尤为重要的.(一语中的) 二.常用方法 1.os.name 输出字符串指示正在使用的平台.如果是windo ...
- html 视频播放器
html 视频播放器 <html> <script> /** *视频播放 *参数说明 u - 媒体URL w - 媒体宽度width h - 媒体高度height */ // ...
- 2.5 webpack 进阶
配置分离 code splitting 异步加载 理解 webpack chunk webpack 调试 2.5.1 配置分离 在大型项目中,可能 webpack.config.js 会变得越来越臃肿 ...
- 前端(十五)—— JavaScript事件:绑定事件方式、事件的冒泡和默认事件、鼠标事件、键盘事件、表单 事件、文档事件、图片事件、页面事件
JS事件:绑定事件方式.事件的冒泡和默认事件.鼠标事件.键盘事件.表单 事件.文档事件.图片事件.页面事件 一.事件的两种绑定方式 1.on事件绑定方式 document.onclick = func ...
- keepalive基础知识
一.LVS负载均衡集群的缺点 二.Keepalived介绍 三.Keepalived的功能 四.Keepalived工作原理 五.Keepalived组件框架 六.Keepalived的安装 6.1 ...