带托盘图标的服务程序,服务运行后为什么没有托盘图标(100)

  • 主题发起人 主题发起人 hnzpeng
  • 开始时间 开始时间
H

hnzpeng

Unregistered / Unconfirmed
GUEST, unregistred user!
网上找的这段代码(带托盘图标的服务程序) 服务安装运行后没有托盘图标? (刚才搜索了一下是 陈经韬先生写的‘用DELPHI创建服务程序’的原代码)program ServiceDemo;uses SvcMgr, Unit_Main in 'Unit_Main.pas' {DelphiService: TService}, Unit_FrmMain in 'Unit_FrmMain.pas' {FrmMain};{$R *.RES}begin
Application.Initialize;
Application.CreateForm(TDelphiService, DelphiService);
Application.CreateForm(TFrmMain, FrmMain);
Application.Run;
end.
unit Unit_Main;interfaceuses Windows, Messages, SysUtils, Classes, Graphics, Controls, SvcMgr, Dialogs, Unit_FrmMain;type TDelphiService = class(TService) procedure ServiceContinue(Sender: TService;
var Continued: Boolean);
procedure ServiceExecute(Sender: TService);
procedure ServicePause(Sender: TService;
var Paused: Boolean);
procedure ServiceShutdown(Sender: TService);
procedure ServiceStart(Sender: TService;
var Started: Boolean);
procedure ServiceStop(Sender: TService;
var Stopped: Boolean);
private { Private declarations } public function GetServiceController: TServiceController;
override;
{ Public declarations } end;
var DelphiService: TDelphiService;
FrmMain: TFrmMain;implementation//uses Unit_FrmMain;{$R *.DFM}procedure ServiceController(CtrlCode: DWord);
stdcall;
begin
DelphiService.Controller(CtrlCode);
end;
function TDelphiService.GetServiceController: TServiceController;
begin
Result := ServiceController;
end;
procedure TDelphiService.ServiceContinue(Sender: TService;var Continued: Boolean);
begin
while not Terminateddo
begin
Sleep(10);ServiceThread.ProcessRequests(False);
end;
end;
procedure TDelphiService.ServiceExecute(Sender: TService);
begin
while not Terminateddo
begin
Sleep(10);MessageBeep(0);ServiceThread.ProcessRequests(False);
end;
end;
procedure TDelphiService.ServicePause(Sender: TService;var Paused: Boolean);
begin
Paused := True;
end;
procedure TDelphiService.ServiceShutdown(Sender: TService);
begin
gbCanClose := true;FrmMain.Free;Status := csStopped;ReportStatus();
end;
procedure TDelphiService.ServiceStart(Sender: TService;var Started: Boolean);
begin
Started := True;Svcmgr.Application.CreateForm(TFrmMain, FrmMain);gbCanClose := False;FrmMain.show;//Hide;改成show后,服务运行,桌面上有窗口弹出来。end;
procedure TDelphiService.ServiceStop(Sender: TService;var Stopped: Boolean);
begin
Stopped := True;gbCanClose := True;FrmMain.Free;
end;
end.
unit Unit_FrmMain;interfaceuses Windows, Messages, SysUtils, Variants, Classes, ShellApi, Graphics, Controls, Forms, Dialogs, ExtCtrls, StdCtrls;constWM_TrayIcon = WM_USER + 1234;type TFrmMain = class(TForm) Timer1: TTimer;
Button1: TButton;
procedure FormCreate(Sender: TObject);
procedure FormCloseQuery(Sender: TObject;
var CanClose: Boolean);
procedure FormDestroy(Sender: TObject);
procedure Timer1Timer(Sender: TObject);
procedure Button1Click(Sender: TObject);
private { Private declarations } IconData: TNotifyIconData;
procedure AddIconToTray;
procedure DelIconFromTray;
procedure TrayIconMessage(var Msg: TMessage);
message WM_TrayIcon;
procedure SysButtonMsg(var Msg: TMessage);
message WM_SYSCOMMAND;
//FICON:TIcon;
public { Public declarations } end;
var FrmMain: TFrmMain;
gbCanClose: Boolean;implementationprocedure TFrmMain.AddIconToTray;
begin
ZeroMemory(@IconData, SizeOf(TNotifyIconData));
IconData.cbSize := SizeOf(TNotifyIconData);
IconData.Wnd := Handle;
IconData.uID := 1;
IconData.uFlags := NIF_MESSAGE or NIF_ICON or NIF_TIP;
IconData.uCallbackMessage := WM_TrayIcon;
IconData.hIcon := Application.Icon.Handle;
IconData.szTip := 'Delphi服务演示程序';
Shell_NotifyIcon(NIM_ADD, @IconData);
end;
procedure TFrmMain.FormCreate(Sender: TObject);
begin
FormStyle := fsStayOnTop;SetWindowLong(Application.Handle, GWL_EXSTYLE, WS_EX_TOOLWINDOW);gbCanClose := False;Timer1.Interval := 1000;Timer1.Enabled := True;// SetWindowLong(Application.Handle, GWL_EXSTYLE, WS_EX_TOOLWINDOW);
end;
{$R *.dfm}procedure TFrmMain.FormCloseQuery(Sender: TObject;
var CanClose: Boolean);
begin
CanClose := gbCanClose;
if not CanClose then
begin
Hide;
end;
end;
procedure TFrmMain.FormDestroy(Sender: TObject);
begin
Timer1.Enabled := False;
DelIconFromTray;
end;
procedure TFrmMain.DelIconFromTray;
begin
Shell_NotifyIcon(NIM_DELETE, @IconData);
end;
procedure TFrmMain.SysButtonMsg(var Msg: TMessage);
begin
if (Msg.wParam = SC_CLOSE) or (Msg.wParam = SC_MINIMIZE) then
Hide else
inherited;
// 执行默认动作end;
procedure TFrmMain.TrayIconMessage(var Msg: TMessage);
begin
if (Msg.LParam = WM_LBUTTONDBLCLK) then
Show();
end;
procedure TFrmMain.Timer1Timer(Sender: TObject);
begin
AddIconToTray;
end;
procedure SendHokKey;stdcall;varHDesk_WL: HDESK;
begin
HDesk_WL := OpenDesktop('Winlogon', 0, False, DESKTOP_JOURNALPLAYBACK);
if (HDesk_WL <> 0) then
if (SetThreadDesktop (HDesk_WL) = True) then
PostMessage(HWND_BROADCAST, WM_HOTKEY, 0, MAKELONG (MOD_ALT or MOD_CONTROL, VK_DELETE));
end;
procedure TFrmMain.Button1Click(Sender: TObject);vardwThreadID : DWORD;
begin
CreateThread(nil, 0, @SendHokKey, nil, 0, dwThreadID);
end;
end.
用原来问题的位置,原来的问题没有解决,是用Timer 做了一个自动关机程序,系统锁定后,关不了机有说要用线程,现在用线程这样做: procedure TcloseThread.Execute;
begin
while now() < Unit1.tido
//ti是关机时间 begin
sleep(10000);
if now() > Unit1.ti then
begin
if Operate='close' then
begin
Form1.closecomputer(EWX_POWEROFF);
end else
if Operate='logout' then
Form1.closecomputer(EWX_LOGOFF) else
if Operate='repeat' then
Form1.closecomputer(EWX_REBOOT);
//调用主线程关机过程 //调用主线程重起过程 //调用主线程注销过程 end;
end;
系统锁定后一样关不了机,不锁定可以关机那位神仙指点迷津
 
用线程?也许是没有权限导致的,可以研究一下OpenProcessToken,LookupPrivilegeValue等函数,对了,我说的是也许
 
系统不锁定都可以正常关机,锁定后就关不了了,根权限有关系?
 
谁知道吗?
 
怎样在系统锁定后还能关机,请神仙指点?
 
高手在哪里?没人知道?
 
没有神仙没有高手结贴散分吧。
 
做成服务可以
 
to 程序开发做成服务后,要安装、卸载吗?
 
那肯定的
 
to 程序开发能不能大至说一下怎么做法,给个例子最好程序启动服务运行,程序退出服务卸载,下次重起程序不运行服务不能启动
 
给你点思路哈程序启动时检测服务安装否,没有就安装,然后启动,退出时停止和卸载服务
 
1.interactive选项2.关机须提升权限,例子本bbs里面多得要死
 
普通应用程序提权没有用
 
to wql interactive 为 true要不然服务启动时不会有窗口弹出来关机函数代码是这样的:procedure TForm1.closecomputer(RebootParam:Longword);var hToken:THANDLE;
tkp:TOKEN_PRIVILEGES;
Nothing:Cardinal;
begin
if Tclose<>nil then
Tclose.Terminate;
CLOSEFG:=true;
OpenProcessToken(GetCurrentProcess, TOKEN_ADJUST_PRIVILEGES or TOKEN_QUERY, hToken);
LookupPrivilegeValue(nil, 'SeShutdownPrivilege', tkp.Privileges[0].Luid);
tkp.PrivilegeCount:=1;
tkp.Privileges[0].Attributes:=SE_PRIVILEGE_ENABLED;
AdjustTokenPrivileges(hToken, FALSE, tkp, 0, nil, Nothing);
ExitWindowsEx(RebootParam, 0);
end;
网上找的,还有一个,不贴了
 
我把 Unit_FrmMain 单独做出来,运行有图标
 
建议服务程序无交互界面。设计成服务程序和该程序的管理程序(交互界面),2者通过文件或者socket或者其他方式连接。也可以参考控件:CoolTrayIcon,提供了你想要的服务+托盘
 
嘿嘿嘿, 我前段时间到软件公司面试,经理就是让我回来做个Firebird的自动备份服务程序,一开始我理解成要将前台和后台服务做成统一体,结果捣鼓了好几个晚上,未果。当时也在一边看MSDN一遍捣鼓,结果发现MSDN上告诉我说,尽可能不要将服务和界面集成。我觉得还是因为Windows有好几个/层桌面,服务只在其中某个或某几个桌面可见,亦或者是执行环境也不同。最后,还是按照MSDN说的,分割界面和服务,分别创建前台管理器和后台服务。目前的设计是通过文件来交换信息的。我觉得,单独运行你的程序,是在GUI环境里,当以服务启动时,是由SCM在后台环境下启动的。我在服务中直接在服务的执行环境下调用应用程序,是不显示在桌面上的,但任务管理器中,该程序又确实启动执行了。
 
其实,就拿SQLserver2000来说吧(我这只有这个svr版本),就是典型的前台管理器+后台服务。这也是我为什么最终放弃统合的原因。不要想什么一劳永逸的方便门了,提供功能的后台服务设计和调试就是非常麻烦的事情,要不然也早满大街飘后台服务了。哦,看了看你的代码,呵呵,不知你有没有读过VCL的源码,那个什么,服务程序的Application是TserviceApplication的实例,不是TApplication的实例,虽然也能创建出窗体,但也有很多不同。我估计那才是他无法在任务栏里显示托盘的原因,如果你非要整合,那就再包含进Forms单元(将Forms作为uses后的第一个单元),然后调用Forms.Application.Initialize;Forms.Application.CrateForm(xxx,xxx);Forms.Application.Run;之流,当然,位置要调整好,不然SCM有可能找不到从哪个位置开始执行服务。
 
权限不够所导。
 
后退
顶部