动态修改PE文件图标(使用UpdateResource API函数)
PE文件的图标存储在资源文件中,而操作资源要用到的API函数就是UpdateResource
首先我们需要先了解一下ICO格式,参考资料:http://www.moon-soft.com/program/FORMAT/windows/icons.htm
ICO格式不复杂,就是由数据头、数据目录、数据三个部分组成
一个.ico文件中可能含有若干个图标,我们需要将数据目录和数据解析出来。简单地写了个单元
- unit Icons;
- interface
- uses
- Winapi.Windows, System.SysUtils, System.Classes;
- type
- { 用于ICO图标文件 }
- TIconDirEntry = packed record
- bWidth: Byte;
- bHeight: Byte;
- bColorCount: Byte;
- bReserved: Byte;
- wPlanes: Word;
- wBitCount: Word;
- dwBytesInRes: DWORD;
- dwImageOffset: DWORD;
- end;
- PIconDirEntry = ^TIconDirEntry;
- TIconDir = packed record
- idReserved: Word;
- idType: Word;
- idCount: Word;
- idEntries: array [0..0] of TIconDirEntry;
- end;
- PIconDir = ^TIconDir;
- { 用于PE文件中的图标 }
- TGroupIconDirEntry = packed record
- bWidth: Byte;
- bHeight: Byte;
- bColorCount: Byte;
- bReserved: Byte;
- wPlanes: Word;
- wBitCount: Word;
- dwBytesInRes: DWORD;
- nID: Word;
- end;
- TGroupIconDir = packed record
- idReserved: Word;
- idType: Word;
- idCount: Word;
- idEntries: array [0 .. 0] of TGroupIconDirEntry;
- end;
- PGroupIconDir = ^TGroupIconDir;
- { ICO图标文件 }
- TIcoFile = class
- public
- IconStream: TMemoryStream;
- IconDir: PIconDir;
- IconDirSize: DWORD;
- constructor Create; overload;
- constructor Create(const FileName: string); overload;
- destructor Destroy; override;
- // 加载ICO数据
- procedure LoadFromFile(const FileName: string);
- procedure LoadFromStream(Stream: TStream);
- end;
- implementation
- { TIcoFile }
- constructor TIcoFile.Create;
- begin
- IconStream := TMemoryStream.Create;
- IconDir := nil;
- IconDirSize := 0;
- end;
- constructor TIcoFile.Create(const FileName: string);
- begin
- Create;
- LoadFromFile(FileName);
- end;
- destructor TIcoFile.Destroy;
- begin
- FreeMem(IconDir);
- FreeAndNil(IconStream);
- inherited;
- end;
- procedure TIcoFile.LoadFromFile(const FileName: string);
- var
- MS: TMemoryStream;
- begin
- MS := TMemoryStream.Create;
- try
- MS.LoadFromFile(FileName);
- LoadFromStream(MS);
- finally
- FreeAndNil(MS);
- end;
- end;
- procedure TIcoFile.LoadFromStream(Stream: TStream);
- var
- Dir: TIconDir;
- begin
- Stream.Position := 0;
- IconStream.Clear;
- IconStream.CopyFrom(Stream, Stream.Size);
- IconStream.Position := 0;
- IconStream.ReadBuffer(Dir, SizeOf(Dir));
- FreeMem(IconDir);
- IconDirSize := SizeOf(TIconDirEntry) * (Dir.idCount - 1) + SizeOf(TIconDir);
- IconDir := AllocMem(IconDirSize);
- IconStream.Position := 0;
- IconStream.ReadBuffer(IconDir^, IconDirSize);
- end;
- end.
这里要注意一个问题,ICO文件中的TIconDirEntry结构和PE文件中的TGroupIconDirEntry结构是不同的
不同处在最后一个参数,ICO文件中表示的是图像数据偏移地址,是DWORD类型。而PE文件中表示的是图像数据的索引,是WORD类型,少了2个字节
替换图标需要写入2个部分:RT_GROUP_ICON 和 RT_ICON
RT_GROUP_ICON也就是ICO的数据头部分,RT_ICON就是图像数据部分了
数据部分直接写入即可,不用转换什么的。但是数据头需要稍微处理一下,因为前面说了,这个替换过程是把ICO文件写到PE文件中
而他们数据头部分结构略有不同(PE文件中每个数据头比起ICO数据头要少2字节),只用把这里处理下就行了
最后写个函数就可以替换PE文件图标了
- function TMainForm.UpdatePeIcon(IcoFile, PeFile: string): Boolean;
- var
- Ico: TIcoFile;
- I: Integer;
- hRes: THandle;
- GroupIconDir: PGroupIconDir;
- GroupIconDirSize: DWORD;
- Data: TBytes;
- begin
- Result := False;
- hRes := BeginUpdateResource(PChar(PeFile), False);
- if hRes <> 0 then
- begin
- Ico := TIcoFile.Create(IcoFile);
- // TGroupIconDirEntry结构要比TIconDirEntry结构少2字节
- GroupIconDirSize := Ico.IconDirSize - Ico.IconDir^.idCount * 2;
- GroupIconDir := AllocMem(GroupIconDirSize);
- GroupIconDir^.idReserved := 0;
- GroupIconDir^.idType := 1;
- GroupIconDir^.idCount := Ico.IconDir.idCount;
- for I := 0 to Ico.IconDir.idCount - 1 do
- begin
- CopyMemory(@GroupIconDir^.idEntries[I], @Ico.IconDir^.idEntries[I],
- SizeOf(TGroupIconDirEntry));
- // 索引从1开始
- GroupIconDir^.idEntries[I].nID := I + 1;
- // 写入图标数据
- SetLength(Data, Ico.IconDir^.idEntries[I].dwBytesInRes);
- Ico.IconStream.Position := Ico.IconDir^.idEntries[I].dwImageOffset;
- Ico.IconStream.ReadBuffer(Data[0], Length(Data));
- UpdateResource(hRes, RT_ICON, MakeIntResource(I + 1), 0, @Data[0], Length(Data));
- end;
- // 写入 RT_GROUP_ICON
- UpdateResource(hRes, RT_GROUP_ICON, MakeIntResource('MAINICON'), 0, GroupIconDir, GroupIconDirSize);
- FreeMem(GroupIconDir);
- FreeAndNil(Ico);
- EndUpdateResource(hRes, False);
- Result := True;
- end;
- end;
http://blog.csdn.net/aqtata/article/details/7710720
动态修改PE文件图标(使用UpdateResource API函数)的更多相关文章
- 替换应用程序exe图标,主要使用BeginUpdateResource,UpdateResource API函数
替换应用程序exe图标,主要使用的API函数是BeginUpdateResource(),UpdateResource(),EndUpdateResource()来使用自定义的ico文件类替换exe程 ...
- 修改PE文件的入口函数OEP
修改入口函数地址.这个是最省事的办法,在原PE文件中新增加一个节,计算新节的RVA,然后修改入口代码,使其指向新增加的节.当然,如果.text节空隙足够大的话,不用添加新节也可以. BOOL Chan ...
- maven 根据P参数值打包动态修改properties文件中值或一定properties
需求:由于最近开发clover项目 ,没有使用spring,更没有使用任何框架,而使用J2EE的web工程,所以连接ZK和MongoDB.Redis等服务器需用指定properties文件, 而目前公 ...
- Dll注入:修改PE文件 IAT注入
PE原理就不阐述了, 这个注入是PE感染的一种,通过添加一个新节注入,会改变PE文件的大小,将原有的导入表复制到新节中,并添加自己的导入表描述符,最后将数据目录项中指向的导入表的入口指向新节. 步骤: ...
- 用UpdateResource修改EXE文件图标(已修正)
//请自行添加到 Type 处PICONDIRENTRY = ^ICONDIRENTRY;ICONDIRENTRY = packed record bWidth: Byte; bHeight: Byt ...
- 动态修改css文件中,具体的class中的个别属性值。
function setStyleSheetObjCssClassProperty(pStyleSheetObj, pSelectorText, pProperty, pValue) { var pS ...
- BS Web窗体 动态修改WebConfig文件参数及数据库链接串
WebConfig操作帮助类 /// /// ConfigurationOperator 的摘要说明 /// public class ConfigurationOperator : IDisposa ...
- 应用程序加载外部字体文件(使用AddFontResource API函数指定字体)
/* MSDN: Any application that adds or removes fonts from the system font table should notify other w ...
- 打开并锁定一个文件(使用LockFile API函数)
var aHandle : THandle; aFileSize : Integer; aFileName : String; procedure TForm1.Button3Click(Sender ...
随机推荐
- 凡客副总裁崔晓琦离职 曾负责旗下V+商城项目_科技_腾讯网
凡客副总裁崔晓琦离职 曾负责旗下V+商城项目_科技_腾讯网 凡客副总裁崔晓琦离职 曾负责旗下V+商城项目 腾讯科技[微博]乐天2013年09月18日12:44 分享 微博 空间 微信 新浪微博 邮箱 ...
- Swift编程语言学习11—— 枚举全局变量、局部变量与类型属性
全局变量和局部变量 计算属性和属性监视器所描写叙述的模式也能够用于全局变量和局部变量,全局变量是在函数.方法.闭包或不论什么类型之外定义的变量,局部变量是在函数.方法或闭包内部定义的变量. 前面章节提 ...
- Java进阶04 RTTI
链接地址:http://www.cnblogs.com/vamei/archive/2013/04/14/3013985.html 作者:Vamei 出处:http://www.cnblogs.com ...
- Cocos Studio和Cocos2d-x版本对应关系
链接地址:http://www.cocoachina.com/bbs/read.php?tid=182077 可以在cocos2d.cpp中查看2d-x的版本信息. 版本对应列表: Studio2 ...
- javascript每日一练(十二)——运动框架
运动框架 可以实现多物体任意值运动 例子: <!doctype html> <html> <head> <meta charset="utf-8&q ...
- 闲扯 Javascript 03 时钟和QQ延时框
时钟 : 所用到得图片 : 开启定时器 setInterval 间隔型 setTimeout 延时型 停止定时器 clearInterval clearTimeout 效果思路 获取系统时间 D ...
- HDOJ1728 BFS【STL__queue_的应用】
STL__queue_的应用 调用的时候要有头文件: #include<stdlib.h> 或 #include<cstdlib> + #include<queue> ...
- JVM调优总结(十二)-参考资料
能整理出上面一些东西,也是因为站在巨人的肩上.下面是一些参考资料,供大家学习,大家有更好的,可以继续完善:) · Java 理论与实践: 垃圾收集简史 · Java SE 6 HotSpot[tm] ...
- load data 方式导入的数据不可以用binlog日志进行恢复,因为binlog里面不产生insert sql语句。
QQ群里面有人问起这个问题: 用load data 导入数据的时候,在binlog文件中记录的不是insert 语句,这样的话,如果用load data 导入数据,当需要恢复数据库的时候 bi ...
- Tomcat提供的安全机制
Tomcat中安全机制 BASIC认证 <login-config> <auth-method>BASIC</auth-method> </login-c ...