Добавил:
Upload Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:
Методичка для КР по ООП.doc
Скачиваний:
8
Добавлен:
18.04.2019
Размер:
2.47 Mб
Скачать

Примеры перехвата сообщений

В данном приложении очередь сообщений запускается без использования дополнительных классов. Выполняется в режиме отладки. Причина, по которой приложение не реагирует на сообщения системы при запуске в обычном режиме, в настоящее время мне не известна.

using System;

using System.Windows.Forms;

namespace MessageLoop01

{

// A message filter.

public class MyMessageFilter : IMessageFilter

{

long nMess = 0;

public bool PreFilterMessage(ref Message m)

{

nMess++;

Console.WriteLine

(“Processing the messages: {0} - {1}: {2}”,m.Msg,

nMess,

Application.MessageLoop);

return false;

}

}

/// <summary>

/// Summary description for Class1.

/// </summary>

class Class1

{

/// <summary>

/// The main entry Point for the application.

/// </summary>

static void Main(string[] args)

{

Application.AddMessageFilter(new MyMessageFilter());

Application.Run();

}

}

}

В следующем примере объект-представитель класса Form связан с оконной процедурой, в рамках которой и запускается цикл обработки сообщений. Замечательное сочетание консольного приложения с окном формы.

using System;

using System.Windows.Forms;

namespace MessageLoop01

{

// A message filter.

public class MyMessageFilter : IMessageFilter

{

long nMess = 0;

public bool PreFilterMessage(ref Message m)

{

nMess++;

Console.WriteLine

(“Processing the messages: {0} - {1}:{2} > {3}”,m.Msg,m.LParam,m.WParam,nMess);

return false;

}

}

class MyForm:Form

{

public MyForm()

{

if (Application.MessageLoop) Console.WriteLine(“Yes!”);

else Console.WriteLine(“No!”);

Application.AddMessageFilter(new MyMessageFilter());

}

}

/// <summary>

/// Summary description for Class1.

/// </summary>

class Class1

{

/// <summary>

/// The main entry Point for the application.

/// </summary>

[STAThread]

static void Main(string[] args)

{

if (Application.MessageLoop) Console.WriteLine(“Yes!”);

else Console.WriteLine(“No!”);

Application.Run(new MyForm());

}

}

}

Метод WndProc

В классах - наследниках класса Control (в классах Form, Кнопка, Список, …) объявляется виртуальный метод WndProc, который обеспечивает обработку передаваемых приложению сообщений Windows. Переопределение этого метода позволяет задавать специфику поведения создаваемого элемента управления, включая и саму форму.

protected virtual void WndProc(ref Message m);

Parameters

m

The Windows Message to process.

Все сообщения после PreProcessMessage метода поступают к WndProc методу. В сущности, ничего не изменилось! WndProc method в точности соответствует WindowProc функции.

Пример переопределения WndProc

Реакция приложения на передаваемые в оконную процедуру WndProc сообщения Windows определяет поведение приложения. Метод WndProc в приложении может быть переопределён. При его переопределении следует иметь в виду, что для адекватного поведения приложения в среде Windows необходимо обеспечить вызов базовой версии метода WndProc. Всё, что не было обработано в рамках переопределённого метода WndProc, должно обрабатываться в методе базового класса.

Ниже демонстрируется пример переопределения метода WndProc. В результате переопределения приложение “научилось” идентифицировать и реагировать на системное сообщение WM_ACTIVATEAPP, которое передаётся каждому приложению при переводе приложения из активного состояния в пассивное и обратно.

using System;

using System.Drawing;

using System.Windows.Forms;

namespace csTempWindowsApplication1

{

public class Form1 : System.Windows.Forms.Form

{

// Здесь определяется константа, которая содержится в «windows.h» header file.

private const int WM_ACTIVATEAPP = 0x001C;

// Флаг активности приложения.

private bool appActive = true;

[STAThread]

static void Main()

{

Application.Run(new Form1());

}

//===================================================================

public Form1()

{

// Задание свойств формы.

this.Size = new System.Drawing.Size(300,300);

this.Text = “Form1”;

this.Font = new System.Drawing.Font(“Microsoft Sans Serif”,

18F,

System.Drawing.FontStyle.Bold,

System.Drawing.GraphicsUnit.Point,

((System.Byte)(0))

);

}

//===================================================================

//===================================================================

// Вот переопределённая оконная процедура.

protected override void WndProc(ref Message m)

{

// Listen for operating system messages.

switch (m.Msg)

{

// Сообщение под кодовым названием WM_ACTIVATEAPP occurs when the application

// becomes the active application or becomes inactive.

case WM_ACTIVATEAPP:

// The WParam value identifies what is occurring.

appActive = (((int)m.WParam != 0));

// Invalidate to get new text painted.

this.Invalidate();

break;

}

base.WndProc(ref m);

}

//===================================================================

//===================================================================

protected override void OnPaint(PaintEventArgs e)

{

// Стиль отрисовки текста определяется состоянием приложения.

if (appActive)

{

e.Graphics.FillRectangle(SystemBrushes.ActiveCaption,20,20,260,50);

e.Graphics.DrawString(“Application is active”,

this.Font, SystemBrushes.ActiveCaptionText, 20,20);

}

else

{

e.Graphics.FillRectangle(SystemBrushes.InactiveCaption,20,20,260,50);

e.Graphics.DrawString(“Application is Inactive”,

this.Font, SystemBrushes.ActiveCaptionText, 20,20);

}

}

//===================================================================

}

}