windows下,在对使用python的编写的程序进行打包时,常用的工具就是py2exe了,该工具只需要一些脚本就看实现将python程序打包成一个exe文件。并且该exe程序的执行不需要系统预先安装python环境。windows下常见的gui成程序在打包时,都不会缺少如下脚本:
setup( windows = [ { "script": "with_gui.py", "icon_resources": [(1, "myicon.ico")] } ], )
其中icon_resources参数是我们要讨论的重点。
按照上面的代码打包来的exe程序,在windows XP或server 2003系统下,我们可以看到exe的图标。但是当把程序拷贝到vista/win7下时,exe图标确变成了默认的"窗口"图标,无论怎么变换试图模式都使如此。
首先先介绍一下ico文件,这对于理解其解决方案有很大帮助。
ico文件是windows下图片格式,我们看到的文件夹,执行文件等都有不同的图标显示,并且当我们切换视图模式(Thumbnails,Tiles,Icons, List, Details)时,文件的图标会以"不同"尺寸显示,确切的说,应该是不同的图标文件(尺寸亦不同)。ico文件里面可以有多个不同的图标文件以适应不同的视图模式,并且这些图标文件通常按尺寸大小的顺序存放。以windows XP/server 2003下支持的ico尺寸为16x16,32x32和48x48。vista/win7下则最多可以支持256x256。
在py2exe官网上贴出了一个解决方案(付出了血,汗,泪的代价):http://www.py2exe.org/index.cgi/CustomIcons,答案来源于http://stackoverflow.com/questions/525329/embedding-icon-in-exe-with-py2exe-visible-in-vista/6198910#6198910。问题关键在于ico文件中图标的顺序问题,XP和server 2003对顺序要求不高,无论是图标是按尺寸的正序还是倒序都可以正常显示,而然在vista/win7下确只能倒序。如果你使用png2ico工具的工具的话,可以使用如下命令生成myicon.ico文件:
png2ico myicon.ico icon_128x128.png icon_64x64.png icon_48x48.png icon_32x32.png icon_16x16.png
而不是:
png2ico myicon.ico icon_16x16.png icon_32x32.png icon_48x48.png icon_64x64.png icon_128x128.png
要提醒的是png2ico生成的ico文件效果不是太好并且图标文件不能达到256x256,你不得不选择选择像248x248这样的尺寸来代替。推荐一个在线的转换工具ConvertIcon!,只是png2ico可以定义图标的顺序,而ConvertIcon!则是从小到大的顺序存储图标文件。但另一个ico编辑工具可以解决这个问题--Greenfish Icon Editor Pro,该工具不仅可以编辑图标文件,而且可以修改图标文件的顺序。
综上,使用在线工具ConvertIcon!来生成ico文件,然后使用Greenfish Icon Editor Pro来修改不同尺寸图标的细节,并修改图标顺序。
1、添加对话框类声明中字体和颜色变量
public: CFont m_myFont; // 字体对象 COLORREF m_myColor; // 颜色对象
2、在对话框初始化函数OnInitDialog()中对字体和颜色进行初始化
// TODO: 在此添加额外的初始化代码 m_myFont.CreatePointFont(150, _T("华文彩云")); m_myColor = RGB(255, 0, 255);
3、为对话框类添加WM_CTLCOLOR消息响应函数,并在其消息响应函数OnCtlColor()中添加代码
afx_msg HBRUSH OnCtlColor(CDC* pDC, CWnd* pWnd, UINT nCtlColor); // 消息响应函数声明
ON_WM_CTLCOLOR() // 消息映射
HBRUSH CXXXDlg::OnCtlColor(CDC* pDC, CWnd* pWnd, UINT nCtlColor) // 消息响应函数实现 { HBRUSH hbr = CDialog::OnCtlColor(pDC, pWnd, nCtlColor); // TODO: 在此更改 DC 的任何属性 if(pWnd->GetDlgCtrlID()==IDC_STATIC_WEEK) { pDC->SelectObject(&m_myFont); // 设置字体 pDC->SetTextColor(m_myColor); // 设置颜色 } // TODO: 如果默认的不是所需画笔,则返回另一个画笔 return hbr; }
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<title>动态创建纵向列表</title>
<style type="text/css">
a { color: #000; text-decoration: none; }
a:hover { color: #F00; }
#menu { width: 100px; border: 1px solid #CCC; border-bottom:none;}
#menu ul { list-style: none; margin: 0px; padding: 0px; }
#menu ul li { background: #eee; padding: 0px 8px; height: 26px; line-height: 26px; border-bottom: 1px solid #CCC; position:relative; }
#menu ul li ul { display:none; position: absolute; left: 100px; top: 0px; width:100px;border:1px solid #ccc; border-bottom:none; }
#menu ul li.current ul { display:block;}
#menu ul li ul li{text-align:center;} /*设置城市内容居中*/
</style>
<script type="text/javascript">
var provs = { "江西省": ["南昌市", "景德镇", "九江", "鹰潭", "萍乡", "新馀", "赣州", "吉安", "宜春", "抚州", "上饶"],
"福建省": ["福州", "厦门", "莆田", "三明", "泉州", "漳州", "南平", "龙岩", "宁德"],
"河北省": ["石家庄", "邯郸", "邢台", "保定", "张家口", "承德", "廊坊", "唐山", "秦皇岛", "沧州", "衡水"],
"四川省": ["成都市", "自贡市", "攀枝花市", "泸州市", "德阳市", "绵阳市", "广元市", "遂宁市", "内江市", "乐山市", "南充市", "眉山市", "宜宾市", "广安市", "达州市", "雅安市", "巴中市", "资阳市", "阿坝藏族羌族自治州", "甘孜藏族自治州", "凉山彝族自治州"],
"山西省": ["太原市", "大同市", "阳泉市", "长治市", "晋城市", "朔州市", "晋中市", "运城市", "忻州市", "临汾市", "吕梁市"],
"内蒙古": ["呼和浩特市", "包头市", "乌海市", "赤峰市", "通辽市", "鄂尔多斯市", "呼伦贝尔市", "巴彦淖尔市", "乌兰察布市", "兴安盟", "锡林郭勒盟", "阿拉善盟"],
"海南省": ["海口市", "三亚市"], "重庆市": ["重庆"],
"贵州省": ["贵阳市", "六盘水市", "遵义市", "安顺市", "铜仁地区", "黔西南布依族苗族自治州", "毕节地区", "黔东南苗族侗族自治州", "黔南布依族苗族自治州"],
"甘肃省": ["兰州市", "嘉峪关市", "金昌市", "白银市", "天水市", "武威市", "张掖市", "平凉市", "酒泉市", "庆阳市", "定西市", "陇南市", "临夏回族自治州", "甘南藏族自治州"],
"青海省": ["西宁市", "海东地区", "海北藏族自治州", "黄南藏族自治州", "海南藏族自治州", "果洛藏族自治州", "玉树藏族自治州", "海西蒙古族藏族自治州"],
"宁夏自治区": ["银川市", "石嘴山市", "吴忠市", "固原市", "中卫市"]
};
function iniEvent() {
var provUL = document.getElementById("prov");
if (provUL) {
var allli = provUL.getElementsByTagName("li");
for (i = 0; i < allli.length; i++) {
node = allli[i];
node.onmouseover = function () { //鼠标经过时显示层
this.className = "current";
}
node.onmouseout = function () { //鼠标离开时隐藏层
this.className = this.className.replace("current", "");
}
}
}
}
function loadData() {
var provUL = document.getElementById("prov");
var nIndex = 0;
for (var key in provs) {
var provLi = document.createElement("li");
provLi.id = "provLI" + nIndex;
provLi.innerHTML = "<a href='#'>" + key + "</a>";
provUL.appendChild(provLi); //添加省份li
//================添加城市========================
var citys = provs[key];
if (citys.length > 0) {
var cityUL = document.createElement("ul");
var maxLength = 0; //存放最大城市内容的长度,以便后面设置cityUL的最大宽度,达到宽度自适应
for (var i = 0; i < citys.length; i++) {
var ci