阻止系统自动睡眠的小软件,附C#制作过程

原文 http://www.cnblogs.com/h46incon/p/SleepPreventer.html

因为有时下载东西的时候,不想让电脑自动深入睡眠,所以就开启了离开模式。这样不但不节能环保,而且到真正想要睡眠的时候就是一翻蛋疼。

改过自新,关闭了离开模式,同时无操作30分钟后也会进入睡眠模式。但是在下载的时候怎么办呢?反正也是闲着,就写了这东西:

增加了合上盖子时保持唤醒功能,截图懒得换了。

阻止系统自动睡眠的小软件,附C#制作过程阻止系统自动睡眠的小软件,附C#制作过程阻止系统自动睡眠的小软件,附C#制作过程阻止系统自动睡眠的小软件,附C#制作过程

 

第一张是主界面,最小化时会隐藏窗口。后面的是托盘的菜单,托盘会根据不同的设置选择不同的图标。

其中,开启离开模式不需要修改注册表,自然的,也就在软件开着的时候有效。监控模式则是每隔半分钟,就将本次设置告知系统。为啥需要这个选项呢?往后看看实现就知道了。(嗯,发现其实不用这个选项的)

程序在最后面。

 


下面是实现(其实写这个程序的目的之一就是为了学习C#……):

(所谓的)核心代码是调用这个API ( MSDN的介绍) :

1
2
3
EXECUTION_STATE WINAPI SetThreadExecutionState(
  _In_  EXECUTION_STATE esFlags
);

这个API作用是允许程序通知系统在使用某些资源,以阻止系统进入睡眠或关闭显示器。

参数esFlags是以下选项的组合:ES_AWAYMODE_REQUIREDES_CONTINUOUSES_DISPLAY_REQUIREDES_SYSTEM_REQUIRED

按字面意思理解选项即可。其中ES_CONTINUOUS表示在下一次调用该API前,本次设置会一直生效。所以在大部分情况下,加上这个选项的话,只需调用一次API即可。但考虑到可能有别的程序也在调用这个API,因而让本程序的设置失效,所以有了监控模式:每隔一断时间就将设置通知系统。(这个API是针对每个线程而言的,只要这个线程不退出,和CONTINUOUS一起设置的选项就会一直生效)。

若是单独使用ES_CONTINUOUS选项,则会恢复睡眠策略。

 

C# 怎么使用Win32API呢?

1
2
3
4
5
6
7
8
9
using System.Runtime.InteropServices;
// 按照API原型,将类型转换C#的类型声明即可
[DllImport("kernel32.dll")]
static extern uint SetThreadExecutionState(uint esFlags);
// 选项所用到的常数
const uint ES_AWAYMODE_REQUIRED = 0x00000040;
const uint ES_CONTINUOUS = 0x80000000;
const uint ES_DISPLAY_REQUIRED = 0x00000002;
const uint ES_SYSTEM_REQUIRED = 0x00000001;

C#所有的方法变量都必须声明在一个类里。我把这些东西声明到一个叫Public的类。声明后,就可以直接调用该方法了。

 

(不知道起什么子标题……)

该程序可以通过主界面以及托盘弹出的菜单进行选择的设置,所以需要同步这两个地方的状态,Checkbox该自动打钩时打钩,该取消时取消等等。为了写代码时,不需要考虑这些东西,就抽象出了一个Option类(每次起名的时候就痛恨自己英文太差)。通过该类可以设置选项,同时也可以在里面注册一个委托,当选项有变时,会调用这些委托。同时,该类也会负责在设置选项时自动通知系统。注:在析构函数中,以参数ES_CONTINUOUS调用一次该API恢复原来的休眠配置。

 

主界面:

    主界面就是拉拉控件,处理下事件。注意,Checkbox的选项值可能会因为鼠标点击以外的原因改变,所以选择监听鼠标点击事件。为了处理事件时少粘贴点代码,我用一个Dictionary将 每个CheckBox和一个选项值绑定在一起。然后使用同一个事件处理函数,在函数里,根据sender确定是哪个Checkbox发送的,再根据那个 Dictionary确定需要设置什么样的值。 然后向Option类注册一个委托,在选项有变时改变Checkbox到正确的状态。

    最小化时直接隐藏窗口,而不是缩小的任务栏:可以选择监听Resize事件,在事件中判断窗体是不是处于最小化的状态,是的话就隐藏窗体。更彻底的是重载窗体类的消息处理函数,并自己处理最小化消息。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
protected override void WndProc(ref Message m)
{
    const int WM_SYSCOMMAND = 0x112;
    const int SC_CLOSE = 0xF060;
    const int SC_MINIMIZE = 0xF020;
    const int SC_MAXIMIZE = 0xF030;
    if (m.Msg == WM_SYSCOMMAND)
    {
        if (m.WParam.ToInt32() == SC_MINIMIZE)
        {
            this.Hide();
            return;
        }
    }
    base.WndProc(ref m);
}

 

托盘部分:

    托盘:使用NotifyIcon控件。注意,这个控件必须设置ICON才能显示。添加一个ContextMenuStrip对 象作为右键点击时的弹出菜单。为了让点击托盘时能弹出这个菜单,可以将托盘控件的ContextMenuStrip属性设置为该菜单即可。在这里因为我需 要监听鼠标事件,让左键点击时显示主窗口,我就在事件处理函数中顺便处理右键点击了。如果是右键点击,则调用该菜单的Show方法就可以了。

    然后是菜单的内容。新建若干ToolStripMenuItem对象,并用ContextMenuStrip.Items.Add(…)方法将这些Item添加进菜单即可。每个Item可以监听鼠标点击事件,同时也可以通过设置它的Checked属性来显示item文本前的小钩钩。需要分割线的话添加ToolStripSeparator对象就可以了。

    动态的托盘图标:为了少写点什么读取文件之类的代码,就直接把托盘图标添加进程序的资源文件里了。方法是在项目的属性里选择资源,然后就可以添加想要的资源了。需要访问这些资源时,在myProject.Properties.Resources里就可以访问了。如:System.Drawing.Bitmap bitmap1 = myProject.Properties.Resources.Image01(参考MSDN)。最后再根据选项的状态选择相应的图标即可。

    托盘消失:直接执行Application.Exit()的话,托盘不会自动消失,得鼠标从托盘上面滑过才可以。为了让托盘显示,执行NotifyIcon对象的Dispose( )方法就好了。

 

让程序开始运行时只显示托盘,而不显示主窗体放狗搜索的时候发现这个问题让很多新手喝了一壶。我的解决方法是:让程序开始是不运行主窗体的代码,只运行托盘控件的代码。转到Main函数,发现Main函数最后执行的是 Application.Run(…) 这个方法。最开始尝试在这个方法的参数里填入一个NotifyIcon对象,但是不行。然后试着直接用new新建一个NotifyIcon对象,这个时候托盘是会显示出来,但是程序马上就会退出了。最后发现这个方法有个无参重载版本,执行后程序就不会退出了……

 

生成的EXE的图标:在项目的属性里,选择应用程序选项卡,就可以设置图标了。


睡眠终结者: http://files.cnblogs.com/h46incon/SleepPreventer.zip

上一篇:包括”/“排除”设置禁用了加载功能。


下一篇:组件通信