Не могу поймать сообщения сна / приостановки (winXP) - PullRequest
6 голосов
/ 07 апреля 2009

Мое приложение должно блокировать спящий режим / режим гибернации. У меня есть код, но после успешного перехвата сообщения WM_POWERBROADCAST ни PBT_APMQUERYSUSPEND , ни PBT_APMQUERYSTANDBY успешно не перехватываются. Интересно, что PBT_APMRESUMECRITICAL и PBT_APMRESUMEAUTOMATIC сообщений перехватываются моим приложением.

Суть вопроса: есть ли причина, по которой мое приложение не может перехватить сообщения ожидания / приостановки, но преуспеть в перехвате сообщений возобновления?

Это Q & A [stackoverflow.com] помогло, между прочим, но, опять же, сообщения, похоже, не попадают в мое приложение.

Мой код (с кратким описанием журнала регистрации событий):

        protected override void WndProc(ref System.Windows.Forms.Message m)
    {
        // Power status event triggered
        if (m.Msg == (int)NativeMethods.WindowMessage.WM_POWERBROADCAST)
        {
            // Machine is trying to enter suspended state
            if (m.WParam.ToInt32() == (int)NativeMethods.WindowMessage.PBT_APMQUERYSUSPEND ||
                m.WParam.ToInt32() == (int)NativeMethods.WindowMessage.PBT_APMQUERYSTANDBY)
            {
                // Have perms to deny this message?
                if((m.LParam.ToInt32() & 0x1) != 0)
                {
                    // If so, deny broadcast message
                    m.Result = new IntPtr((int)NativeMethods.WindowMessage.BROADCAST_QUERY_DENY);
                }
            }
            return; // ?!
        }

        base.WndProc(ref m);
    }

Ответы [ 5 ]

3 голосов
/ 09 апреля 2009

Теперь работает как для XP, так и для Vista. Я создал заглушку winform app с соответствующим кодом (очевидно, его можно очистить, но он передает смысл).

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
using System.Runtime.InteropServices;

namespace standbyTest
{
    public partial class Form1 : Form
    {

        [DllImport("Kernel32.DLL", CharSet = CharSet.Auto, SetLastError = true)]
        protected static extern EXECUTION_STATE SetThreadExecutionState(EXECUTION_STATE state);

        [Flags]
        public enum EXECUTION_STATE : uint
        {
            ES_CONTINUOUS = 0x80000000,
            ES_DISPLAY_REQUIRED = 2,
            ES_SYSTEM_REQUIRED = 1,
            ES_AWAYMODE_REQUIRED = 0x00000040
        }

        public Form1()
        {
            if(Environment.OSVersion.Version.Major > 5)
            {
                // vista and above: block suspend mode
                SetThreadExecutionState(EXECUTION_STATE.ES_AWAYMODE_REQUIRED | EXECUTION_STATE.ES_SYSTEM_REQUIRED | EXECUTION_STATE.ES_CONTINUOUS);
            }

            InitializeComponent();

            //MessageBox.Show(string.Format("version: {0}", Environment.OSVersion.Version.Major.ToString() ));

        }

        protected override void OnClosed(EventArgs e)
        {
            base.OnClosed(e);

            if(Environment.OSVersion.Version.Major > 5)
            {
                // Re-allow suspend mode
                SetThreadExecutionState(EXECUTION_STATE.ES_CONTINUOUS);
            }
        }


        protected override void WndProc(ref System.Windows.Forms.Message m)
        {
            // Power status event triggered
            if(m.Msg == (int)WindowMessage.WM_POWERBROADCAST)
            {
                // Machine is trying to enter suspended state
                if(m.WParam.ToInt32() == (int)WindowMessage.PBT_APMQUERYSUSPEND ||
                        m.WParam.ToInt32() == (int)WindowMessage.PBT_APMQUERYSTANDBY)
                {
                    // Have perms to deny this message?
                    if((m.LParam.ToInt32() & 0x1) != 0)
                    {
                        // If so, deny broadcast message
                        m.Result = new IntPtr((int)WindowMessage.BROADCAST_QUERY_DENY);
                    }
                }
                return;
            }

            base.WndProc(ref m);
        }
    }



    internal enum WindowMessage
    {

        /// <summary>
        /// Notify that machine power state is changing
        /// </summary>
        WM_POWERBROADCAST = 0x218,
        /// <summary>
        /// Message indicating that machine is trying to enter suspended state
        /// </summary>
        PBT_APMQUERYSUSPEND = 0x0,
        PBT_APMQUERYSTANDBY = 0x0001,

        /// <summary>
        /// Message to deny broadcast query
        /// </summary>
        BROADCAST_QUERY_DENY = 0x424D5144


    }
}
1 голос
/ 11 октября 2010
0 голосов
/ 08 апреля 2009

Я попробовал этот же код в тестовом приложении на моей (dev) машине и на другой (тестовой) машине (также winXP). На моей машине она продолжает выходить из строя, то есть машина уходит в сон. Но на другой машине все работает! Сначала я подумал, что это проблема отладки и режима выпуска, но это не так.

Кажется, что-то другое в моей машине разработчика, хотя я понятия не имею, что это может быть.

Тайна разгадана ... Сорта.

0 голосов
/ 08 апреля 2009

В Vista вызовите SetThreadExecutionState, чтобы уведомить WPM о том, что система не находится в режиме ожидания.

В Windows XP / 2000:
Приложение может вернуть BROADCAST_QUERY_DENY, чтобы отклонить запрос PBT_APMQUERYSUSPEND или PBT_APMQUERYSUSPENDFAILED.

MSDN: Windows XP и более ранние версии: система передает событие PBT_APMQUERYSUSPEND, чтобы запросить разрешение на приостановку работы системы. Система ожидает, что каждое приложение и драйвер определят, должно ли запрошенное событие произойти, и вернут TRUE, если оно произойдет, или вернут BROADCAST_QUERY_DENY в противном случае. Приложения не должны отклонять этот запрос. Если приложение отклоняет этот запрос, система передает событие PBT_APMQUERYSUSPENDFAILED. Это событие уведомляет приложения и драйверы о продолжении работы в обычном режиме.

Также я не думаю, что PBT_APMQUERYSTANDBY или PBT_APMSTANDBY поддерживаются в Win2K. Вы пробовали глобально регистрировать трансляцию, когда Windows выключается, чтобы увидеть, отправляются ли они?

0 голосов
/ 07 апреля 2009

Вы работаете в Vista или Windows Server 2008? Эта страница говорит

Из-за изменений в модели управления питанием для Windows Vista и Windows Server 2008 событие PBT-APMQUERYSUSPEND больше не доставляется приложениям. Вместо этого доставляется событие BT_APMSUSPEND ...

Может быть, поэтому ты этого не видишь?

...