• 全部
  • VC综合技术
  • 互联网技术
  • MFC AppLauncher
  • .NET 技术
  • 界面
  • 进程
  • 算法
  • 硬件/系统
  • 数据库
  • VC++技术资源

基于单文档或对话框最小化时,程序如何放在System Tray中(在线恭候!)

lazyseep 2002-05-14 10:43:41
如题,谢谢!!
有源代码就给满分。
...全文
6 点赞 收藏 9
写回复
9 条回复
切换为时间正序
当前发帖距今超过3年,不再开放新的回复
发表回复
Anaki 2002-05-14
已发出,注意查收...
回复
xuying 2002-05-14
利用任务栏上的图标与用户交互

作者:李奔


我们有时需要编制一些仅在后台监控的程序,为了不干扰前台程序的运行界面和不显示不必要的窗口,应使其运行时的主窗口不可见。同时,应该让用户知道该程序正在运行,并且达到与用户进行交互的目的。将一个图标显示在任务栏右端静态通告区中并响应用户的鼠标动作是当前非常流行的方法,它体现了Windows 95友好的界面风格。下面以一个SDI(单文档界面)程序为例,讲述采用Microsoft Visual C++ 5.0开发这类程序的主要步骤。

首先,要使程序的主窗口不可见,并且不在任务栏上出现任务按钮,要做到这两点,需分别设置主边框窗口的风格和扩展风格:

BOOL CMainFrame::PreCreateWindow(CREATESTRUCT& cs)

{

cs.style =WS_POPUP;//使主窗口不可见

cs.dwExStyle |=WS_EX_TOOLWINDOW;//不显示任务按钮

return CFrameWnd::PreCreateWindow(cs);

}

其次,利用系统函数Shell_NotifyIcon将一个图标显示在任务栏的通告区中。该函数的原型为:

WINSHELLAPI BOOL WINAPI Shell_NotifyIcon(

DWORD dwMessage,

PNOTIFYICONDATA pnid

);

下例中被显示的是主边框窗口的图标,实际上可以显示任何图标:

int CMainFrame::OnCreate(LPCREATESTRUCT lpCreateStruct)

{



NOTIFYICONDATA tnd;

tnd.cbSize=sizeof(NOTIFYICONDATA);

tnd.hWnd=this->m_hWnd;

tnd.uID=IDR_MAINFRAME;

tnd.uFlags=NIF_MESSAGE|NIF_ICON|NIF_TIP;

tnd.uCallbackMessage=WM_LIBEN;

tnd.hIcon=LoadIcon(AfxGetInstanceHandle(),MAKEINTRESOURCE(IDR_MAINFRAME));

strcpy(tnd.szTip,"提示信息");

Shell_NotifyIcon(NIM_ADD,&tnd);



}

在调用该函数之前,需要确定其参数的取值,其中之一为一个具有NOTIFYICONDATA类型的结构。其原型为:

typedef struct _NOTIFYICONDATA { // nid

DWORD cbSize;

HWND hWnd;

UINT uID;

UINT uFlags;

UINT uCallbackMessage;

HICON hIcon;

charszTip[64]; }

NOTIFYICONDATA, *PNOTIFYICONDATA;

在该结构的成员中,cbSize为该结构所占的字节数,hWnd为接受该图标所发出的消息的窗口的句柄,uID为被显示图标的ID,uFlags指明其余的几个成员(hIcon、uCallBackMessage和szTip)的值是否有效,uCallbackMessage为一个自定义的消息,当用户在该图标上作用一些鼠标动作时,将向hWnd成员中指定的窗口发出该消息,可以定义该消息为WM_USER+100。hIcon为被显示图标的句柄,szTip为一字符数组,当鼠标停留在该图标上时,将其内容显示在浮动的提示信息框中。Shell_NotifyIcon函数的另一个参数是一个预定义的消息,可以取如下值之一:NIM_ADD、NIM_DELETE或NIM_MODIFY,分别表示添加图标、删除图标或修改图标。

最后,要与用户进行交互,也就是当用户在该图标上单击或双击鼠标左键或右键时要执行相应的操作,至少也要响应用户终止该程序的意愿。上面已经提到,当用户在图标上进行鼠标动作时,将向hWnd成员中指定的窗口发出自定义的消息,该消息由uCallbackMessage成员指定(在上例中为WM_LIBEN,取值为WM_USER+100)。因此,我们的任务就是在hWnd窗口中响应该自定义消息:

void CMainFrame::OnLiben(WPARAM wParam,LPARAM lParam)

{

UINT uID;//发出该消息的图标的ID

UINT uMouseMsg;//鼠标动作

POINT pt;

uID=(UINT) wParam;

uMouseMsg=(UINT) lParam;

if(uMouseMsg==WM_RBUTTONDOWN)//如果是单击右键

{

switch(uID)

{

case IDR_MAINFRAME://如果是我们的图标

GetCursorPos(&pt);//取得鼠标位置

…//执行相应操作

break;



default:



}

}

return;

}

需要注意的是,首先要在该窗口类的头文件中给出该消息映射函数的原型说明:

afx_msg void OnLiben(WPARAM wParam,LPARAM lParam);

并且要在CPP文件中的消息映射中加入相应的条目,注意一定要加在//{{AFX_MSG_MAP(CMainFrame)和//}}AFX_MSG_MAP之外:

BEGIN_MESSAGE_MAP(CMainFrame, CFrameWnd)

//{{AFX_MSG_MAP(CMainFrame)

ON_WM_CREATE()

ON_COMMAND(ID_APP_EXIT, OnAppExit)

//}}AFX_MSG_MAP

ON_MESSAGE(WM_LIBEN,OnLiben)

END_MESSAGE_MAP()

当程序结束时,需要删去通告区中的图标,这时同样应该调用Shell_NotifyIcon函数,只不过第一个参数是表示删除图标的NIM_DELETE了:

void CMainFrame::OnAppExit()

{

// TODO: Add your command handler code here

NOTIFYICONDATA tnid;

tnid.cbSize=sizeof(NOTIFYICONDATA);

tnid.hWnd=this->m_hWnd;

tnid.uID=IDR_MAINFRAME;//保证删除的是我们的图标

Shell_NotifyIcon(NIM_DELETE,&tnid);

AfxPostQuitMessage(0);

}

通过类似的步骤,读者可以响应其他的消息,完成更加高级的交互功能,这里不再赘述。上文所述是笔者经验所得,肯定有不到之处,欢迎指正。

回复
lazyseep 2002-05-14
例程,能发到我的邮箱haiquan@comeinfo.com吗?
谢谢!
回复
lazyseep 2002-05-14
我试一试……
回复
Anaki 2002-05-14
你也可以发消息给我索取例程...
回复
野男孩 2002-05-14
Shell_NotifyIcon()
回复
Anaki 2002-05-14
添加自定义消息
///////////////////////// xx.h
...
#define WM_USER_TRAYNOTIFY WM_APP + 8048 // 自定义消息 ID
...
// Generated message map functions
//{{AFX_MSG(CToggleTopMainDlg)
...
//}}AFX_MSG
afx_msg void OnTrayNotify(WPARAM wParam, LPARAM lParam);
DECLARE_MESSAGE_MAP()

//////////////////////////////////////////// xx.cpp
...
BEGIN_MESSAGE_MAP(CToggleTopMainDlg, CDialog)
//{{AFX_MSG_MAP(CToggleTopMainDlg)
...
//}}AFX_MSG_MAP
ON_MESSAGE(WM_USER_TRAYNOTIFY, OnTrayNotify)
END_MESSAGE_MAP()
...
void Cxx::OnTrayNotify(WPARAM wParam, LPARAM lParam)
{
UINT uID; //发出消息的图标ID
UINT uMouseMsg; //鼠标消息

uID=(UINT) wParam;
uMouseMsg=(UINT) lParam;
if(uMouseMsg == WM_RBUTTONUP) // 右击
{
if(uID == IDR_MAINFRAME) // 是自己的图标 {
...
}
}
if(uMouseMsg == WM_LBUTTONDBLCLK && //左双击
uID == IDR_MAINFRAME ) //是自己的图标
{
...
}

return;
}
回复
lazyseep 2002-05-14
请问,代码是放在OnCreate中还是放在PreCreateWindow中,然后,WM_USER_TRAYNOTIFY怎么定义?
回复
Anaki 2002-05-14
// 我用的一段代码
NOTIFYICONDATA m_struNID
m_struNID.cbSize=sizeof(NOTIFYICONDATA);
m_struNID.hWnd=this->m_hWnd;
m_struNID.uID=IDR_MAINFRAME;
m_struNID.uFlags=NIF_MESSAGE|NIF_ICON|NIF_TIP;
m_struNID.uCallbackMessage=WM_USER_TRAYNOTIFY;
m_struNID.hIcon=LoadIcon(AfxGetInstanceHandle(), MAKEINTRESOURCE(IDR_MAINFRAME));
_tcscpy(m_struNID.szTip, _T("ToggleTop"));

Shell_NotifyIcon(NIM_ADD, &m_struNID);

具体信息可以查看 MSDN 中 对 API Shell_NotifyIcon 的描述
回复
相关推荐
发帖
VC/MFC
创建于2007-09-28

1.5w+

社区成员

VC/MFC相关问题讨论
申请成为版主
帖子事件
创建了帖子
2002-05-14 10:43
社区公告

        VC/MFC社区版块或许是CSDN最“古老”的版块了,记忆之中,与CSDN的年龄几乎差不多。随着时间的推移,MFC技术渐渐的偏离了开发主流,若干年之后的今天,当我们面对着微软的这个经典之笔,内心充满着敬意,那些曾经的记忆,可以说代表着二十年前曾经的辉煌……
        向经典致敬,或许是老一代程序员内心里面难以释怀的感受。互联网大行其道的今天,我们期待着MFC技术能够恢复其曾经的辉煌,或许这个期待会永远成为一种“梦想”,或许一切皆有可能……
        我们希望这个版块可以很好的适配Web时代,期待更好的互联网技术能够使得MFC技术框架得以重现活力,……