将下面的代码拷贝到一个单元中,创建一个包,加入这个单元后安装.使用的时候设置好背景颜色,边框颜色,图标(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风格按钮控件的更多相关文章

  1. 使用GDI+绘制的360风格按钮控件(使用CN_DRAWITEM消息重绘,并使用TGPGraphics,TGPPen,TGPImage,TGPBitmap等)good

    将下面的代码拷贝到一个单元中,创建一个包,加入这个单元后安装.使用的时候设置好背景颜色,边框颜色,图标(png格式)相对路径的文件名称.这个控件可以利用PNG图像的颜色透明特性,背景色默认透明度为50 ...

  2. MFC基于对话框风格按钮控件添加图片的方法(大神止步)

    菜鸟还在研究这个东西,大神就不要看了.一直都在觉得用VC或VS建立的对话框总是全灰色感觉太单调了,如果可以在上面添加一些漂亮的图片就好了,今天终于实现了.其实挺简单的,下面就分几个步骤讲一下: 第一步 ...

  3. dskinlite(uieasy mfc界面库)使用记录2:绘制动态元素(按钮控件绘制元素动态控制,改变图片和文字)

    效果图:这4个分别是按钮按下后4种状态的效果 第88行是显示默认的按钮文字,没有id,SetWindowText改的就是它了 第87行是左边的图片,id是ico,可以通过程序控制 第89行是蓝色的文字 ...

  4. dskinlite(uieasy mfc界面库)使用记录3:绘制动态元素(按钮控件通过隐藏方式修改图片显示)

    效果图: 分别是:正常,正常鼠标悬停,按下,按下鼠标悬停 XML代码: 75,76行定义了一个image,注意id和index属性 初始化代码: click代码: 147,148,153,154:通过 ...

  5. Qt编写自定义控件11-设备防区按钮控件

    前言 在很多项目应用中,需要根据数据动态生成对象显示在地图上,比如地图标注,同时还需要可拖动对象到指定位置显示,能有多种状态指示,安防领域一般用来表示防区或者设备,可以直接显示防区号,有多种状态颜色指 ...

  6. Qt编写自定义控件9-导航按钮控件

    前言 导航按钮控件,主要用于各种漂亮精美的导航条,我们经常在web中看到导航条都非常精美,都是html+css+js实现的,还自带动画过度效果,Qt提供的qss其实也是无敌的,支持基本上所有的CSS2 ...

  7. WinRT自定义控件第一 - 转盘按钮控件

    之前的文章中,介绍了用WPF做一个转盘按钮控件,后来需要把这个控件移植到WinRT时,遇到了很大的问题,主要原因在于WPF和WinRT还是有很大不同的.这篇文章介绍了这个移植过程,由于2次实现的控件功 ...

  8. MFC编程入门之二十二(常用控件:按钮控件Button、Radio Button和Check Box)

    本节继续讲解常用控件--按钮控件的使用. 按钮控件简介 按钮控件包括命令按钮(Button).单选按钮(Radio Button)和复选框(Check Box)等.命令按钮就是我们前面多次提到的侠义的 ...

  9. C#自定义Button按钮控件

    C#自定义Button按钮控件 在实际项目开发中经常可以遇到.net自带控件并不一定可以满足需要,因此需要自定义开发一些新的控件,自定义控件的办法也有多种,可以自己绘制线条颜色图形等进行重绘,也可以采 ...

随机推荐

  1. react 脚手架使用

    下载脚手架 npm i react create-react-app //下载安装脚手架 create-react-app pro_nmae //创建一个react项目 尽量不要使用大写,因为在Lin ...

  2. MySQL不支持事务处理的解决方法

    MySQL数据库默认的存储引擎类型是MyISAM,这种存储引擎类型不支持事务处理. 在MySQL中,只有InnoDB存储引擎类型的数据表才能支持事务处理. 因此,如果想让MySQL支持事务处理,只要将 ...

  3. Vue学习笔记【18】——Vue中的动画(使用过渡类名)

    为什么要有动画:动画能够提高用户的体验,帮助用户更好的理解页面中的功能: 使用过渡类名 步骤分析  需求: 点击按钮,让 h3 显示,再点击,让 h3 隐藏  1. 使用 transition 元素, ...

  4. Python每日一题 002

    做为 Apple Store App 独立开发者,你要搞限时促销,为你的应用生成激活码(或者优惠券),使用 Python 如何生成 200 个激活码(或者优惠券)? 在此生成由数字,字母组成的20位字 ...

  5. JSP自定义标签(转)

    1. TagSupport与BodyTagSupport的区别 TagSupport与BodyTagSupport的区别主要是标签处理类是否需要与标签体交互,如果不需要交互的就用TagSupport, ...

  6. Dynamic partition strict mode requires at least one static partition column.

    https://blog.csdn.net/huobumingbai1234/article/details/81099856

  7. Jquery的Ready方法加载为什么两次?

    Ready方法会调用两次? 查看对应的页面是否存在<iframe src="#" --> 存在iframe加载这个页面的时候,页面就会加载两次. $(document) ...

  8. Java List T 去掉重复对象-java8

    Stream语法详解 Stream当成一个高级版本的Iterator.原始版本的Iterator,用户只能一个一个的遍历元素并对其执行某些操作:高级版本的Stream,用户只要给出需要对其包含的元素执 ...

  9. PAT_A1081#Rational Sum

    Source: PAT A1081 Rational Sum (20 分) Description: Given N rational numbers in the form numerator/de ...

  10. eduCF#61 C. Painting the Fence /// DP 选取k段能覆盖的格数

    题目大意: 给定n m 接下来给定m个在n范围内的段的左右端 l r 求选取m-2段 最多能覆盖多少格 #include <bits/stdc++.h> using namespace s ...