Подглядываем через веб-камеру: учимся использовать встроенную веб-камеру в своих целях

Некоторые люди опасаются, что встроенная в их ноутбуки вебкамера может за ними подглядывать. Они опасаются этого настолько серьезно, что порой даже заклеивают изолентой ее недремлющее око. Делают они это не зря. Мы расскажем, как можно программно овладеть встроенной в ноутбук веб-камерой и использовать ее функционал как в мирных, так и не очень целях.

Начинаем реализацию: первые досадные огорчения

Я был очень удивлен и расстроен, когда узнал, что в великом и могучем .NET Framework напрочь отсутствует возможность простого взаимодействия с веб-камерами. В четвертой версии ситуация улучшилась (для SilverLight-проектов точно появились соответствующие классы), но протестировать я их не успел, поскольку пример для данной статьи я начал писать еще до официального выхода VS2010 и 4-го .NET’a.

Практически отчаявшись, я плотно засел в гугле. Результаты поиска по рунету меня не вдохновили – все, что я нашел – это ссылки на MSDN и технологию DirectDraw. Я даже попробовал набросать простенький примерчик, но из-за отсутствия опыта работы с DirectDraw меня постиг облом. У меня получилось собрать совсем простенькое приложение, но я так и не смог выловить в нем все глюки.

Еще больше отчаявшись, я принялся шерстить ресурсы наших западных товарищей. Проштудировав несколько десятков ссылок, я смог нарыть много вкусностей. Среди них были всевозможные примеры и небольшие статейки (американцы не любят много писать). Мне даже удалось найти рабочий пример на основе DirectDraw, но, когда я увидел код – ужаснулся. Разобраться в нем было тяжело. Поэтому я решил с ним не заморачиваться, а попытаться найти способ попроще. Не успел я распрощаться с примером на DirectDraw, как на глаза мне попался еще один. Автор примера закодил целую библиотеку для работы с веб-камерами и другими устройствами видеозахвата, используя технологию VFW (Video For Windows).

Жаль, что проект автора (я про библиотеку) был максимально кастрирован. Все, что позволяла сделать библиотека – вывести изображение с веб-камеры. Ни захвата отдельных кадров, ни записи видео и других полезных нам фич не было.

И тем не менее, мое подсознание решительно сказало мне, что этот проект и есть то, что я искал. Не успел я беглым взглядом пробежаться по его коду, как увидел имена знакомых win-сообщений и не менее знакомых названий WinAPI функций. Когда-то давным-давно мне приходилось писать приложение для работы с веб-камерой на Delphi. Тогда я и столкнулся с этими функциями впервые.

Посмотрев сорцы, я решил написать свою версию библиотеки и снабдить ее нужным функционалом.

Взвод, готовность №1

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

Для решения этой нехитрой задачи в WindowsAPI предусмотрена функция capGetDriverDescription(). Она принимает пять параметров:

  1. wDriverIndex – индекс драйвера видеозахвата. Значение индекса может варьироваться от 0 до 9;
  2. lpszName – указатель на буфер, содержащий соответствующее имя драйвера;
  3. cbName – размер (в байтах) буфера lpszName;
  4. lpszVer – указатель на буфер, содержащий описание определенного драйвера;
  5. cbVer – размер буфера (в байтах), в котором хранится описание драйвера.

В случае успешного выполнения, функция вернет TRUE. Описание функции у нас есть, теперь посмотрим, как определить ее в C#. Делается это так:

[DllImport("avicap32.dll")]
protected static extern bool capGetDriverDescriptionA (short wDriverIndex, [MarshalAs(UnmanagedType.VBByRefStr)] ref String lpszName, int cbName, [MarshalAs(UnmanagedType.VBByRefStr)] ref String lpszVer, int cbVer);

Обрати внимание, что перед тем, как указать имя подключаемой функции, в обязательном порядке требуется написать имя DLL, в которой она определена. В нашем случае это avicap32.dll.

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

public static Device[] GetAllCapturesDevices()
{
String dName = "".PadRight(100);
String dVersion = "".PadRight(100);

for (short i = 0; i < 10; i++)
{
if (capGetDriverDescriptionA(i,
ref dName, 100, ref dVersion,
100))
{
Device d = new Device(i);
d.Name = dName.Trim();
d.Version = dVersion.Trim();

devices.Add(d);
}
}
return (Device[])devices.ToArray
(typeof(Device));
}

Код выглядит проще некуда. Самое интересное место в нем – цикл, в котором происходит вызов упомянутой выше функции capGetDriverDescription. Из MSDN мы знаем, что индекс (первый параметр функции capGetDriverDescription()) может варьироваться от 0 до 9, поэтому мы целенаправленно запускаем цикл в этом диапазоне. Результатом выполнения метода будет массив классов Device (этот класс я определил самостоятельно, смотри соответствующие исходники).

С получением списка устройств разобрались, теперь позаботимся об отображении видеопотока с камеры. Тут нам сослужит хорошую службу функция capCreateCaptureWindow(), предназначенная для создания окна захвата.

Немного забегая вперед, скажу, что дальнейшие действия с камерой будут происходить путем банальной отправки сообщений окну захвата. Да, именно так, придется воспользоваться до боли знакомой windows-программисту (и приколисту) функцией SendMessage().

Теперь присмотримся внимательнее к функции capCreateCaptureWindow(). Ей требуется передать шесть аргументов:

  1. lpszWindowName – нуль-терминальная строка, содержащая имя окна захвата;
  2. dwStyle – стиль окна;
  3. x – координата X;
  4. y – координата Y;
  5. nWidth – ширина окна;
  6. nHeight – высота окна;
  7. hWnd – handle родительского окна;
  8. nID – идентификатор окна.

Результатом выполнения функции будет handle созданного окна или NULL в случае ошибки. Поскольку эта функция также относится к WinAPI, то ее опять-таки нужно импортировать. Код импортирования приводить не буду, поскольку он практически идентичен тому, что я писал для функции capGetDriverDescription(). Лучше сразу взглянем на процесс инициализации камеры:

deviceHandle = capCreateCaptureWindowA (ref deviceIndex, WS_VISIBLE | WS_CHILD, 0, 0, windowWidth, windowHeight, handle, 0);

if (SendMessage(deviceHandle, WM_CAP_DRIVER_CONNECT, this.index, 0) > 0)
{
SendMessage(deviceHandle, WM_CAP_SET_SCALE, -1, 0);
SendMessage(deviceHandle, WM_CAP_SET_PREVIEWRATE, 0x42, 0);
SendMessage(deviceHandle, WM_CAP_SET_PREVIEW, -1, 0);

SetWindowPos(deviceHandle, 1, 0, 0, windowWidth, windowHeight, 6);
}

В этом коде сразу после создания окна производится попытка отправки сообщения WM_CAP_DRIVER_CONNECT. Отличный от нуля результат выполнения функции расскажет нам о ее успешности.

Теперь представим, что сегодня боги на нашей стороне и произведем незамедлительную отправку нескольких сообщений: WM_CAP_SET_SCALE, WM_CAP_SET_PREVIEWRATE, WM_CAP_SET_PREVIEW. Увы, как и в случае с функциями, C# ничего не знает о существовании этих констант. Тебе опять придется определять их самостоятельно. Список всех необходимых констант с комментариями я привел ниже.

//Пользовательское сообщение
private const int WM_CAP = 0x400;
//Соединение с драйвером устройства видеозахвата
private const int WM_CAP_DRIVER_CONNECT = 0x40a;
//Разрыв связи с драйвером видеозахвата
private const int WM_CAP_DRIVER_DISCONNECT = 0x40b;
//Копирование кадра в буффер обмена
private const int WM_CAP_EDIT_COPY = 0x41e;
//Включение/отключение режима предпосмотра
private const int WM_CAP_SET_PREVIEW = 0x432;
//Включение/отключение режима оверлей
private const int WM_CAP_SET_OVERLAY = 0x433;
//Скорость previewrate
private const int WM_CAP_SET_PREVIEWRATE = 0x434;
//Включение/отключение масштабирования
private const int WM_CAP_SET_SCALE = 0x435;
private const int WS_CHILD = 0x40000000;
private const int WS_VISIBLE = 0x10000000;
//Установка callback-функции для preview
private const int WM_CAP_SET_CALLBACK_FRAME = 0x405;
//Получение одиночного фрейма с драйвера видеозахвата
private const int WM_CAP_GRAB_FRAME = 0x43c;
//Сохранение кадра с камеры в файл
private const int WM_CAP_SAVEDIB = 0x419;

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

Всего в библиотеке я реализовал (точнее, дописал) пару методов: GetAllDevices (уже рассматривали), GetDevice (получение драйвера устройства видеозахвата по индексу), ShowWindow (отображение изображения с веб-камеры), GetFrame (захват отдельного кадра в графический файл) и GetCapture (захват видеопотока).

В качестве демонстрации работоспособности изготовленной либы я набросал небольшое приложение. На форме я расположил один компонент ComboBox (используется для хранения списка имеющихся устройств видеозахвата) и несколько кнопок – "Обновить", "Пуск", "Остановить" и "Скриншот". Ах да, еще на моей форме пестреет компонент Image. Его я применяю для отображения видео с камеры.

Разбор полетов начнем с кнопки "Обновить". По ее нажатию я получаю список всех установленных устройств видеозахвата. Начинка этого обработчика события:

Device[] devices = DeviceManager.GetAllDevices();
foreach (Device d in devices)
{
cmbDevices.Items.Add(d);
}

Правда, все просто? Разработанная нами библиотека берет на себя все черную работу и нам остается лишь наслаждаться объектно-ориентированным программированием. Еще проще выглядит код для включения отображения видеопотока с камеры:

Device selectedDevice = DeviceManager.GetDevice(cmbDevices.SelectedIndex);
selectedDevice.ShowWindow(this.picCapture);

Опять же, все проще пареной репы. Ну и теперь взглянем на код кнопки "Скриншот":

Device selectedDevice = DeviceManager.GetDevice(cmbDevices.SelectedIndex);
selectedDevice.FrameGrabber();

Я не стал уделять особого внимания методу FrameGrabber(). В моем исходнике вызов метода приводит к сохранению текущего кадра прямо в корень системного диска. Разумеется, это не очень корректно, поэтому перед боевым применением программы не забудь внести все необходимые поправки.

Готовность № 3

Теперь настало время поговорить о том, как соорудить простенькую, но надежную систему видеонаблюдения. Обычно такие системы базируются на двух алгоритмах: различие двух фреймов и простое моделирование фона. Их реализация (код) достаточно объемна, поэтому в самый последний момент я решил пойти по более простому пути. Под легким путем подразумевается использование мощного, но пока малоизвестного фреймворка для .NET – AForge.NET.

AForge.NET в первую очередь предназначен для разработчиков и исследователей. С его помощью, девелоперы могут существенно облегчить свой труд при разработке проектов для следующих областей: нейросети, работа с изображениями (наложение фильтров, редактирование изображений, попиксельная фильтрация, изменение размера, поворот изображения), генетика, робототехника, взаимодействие с видео устройствами и т.д. С фреймворком поставляется хорошая документация. В ней описаны абсолютно все возможности продукта. Не поленись хорошенько с ней ознакомиться. Особенно мне хочется отметить качество кода этого продукта. Все написано цивильно и копаться в коде – одно удовольствие.

Теперь вернемся к нашей непосредственной задаче. Скажу честно, средствами фреймворка она решается как дважды два. "Тогда зачем ты мне парил мозг WinAPI функциями?" – недовольно спросишь ты. А за тем, чтобы ты не был ни в чем ограничен. Сам ведь знаешь, что проекты бывают разные. Где-то удобнее применить махину .NET, а где-то проще обойтись старым добрым WinAPI.

Вернемся к нашей задачке. Для реализации детектора движений нам придется воспользоваться классом MotionDetector из вышеупомянутого фреймворка. Класс отлично оперирует объектами типа Bitmap и позволяет быстренько вычислить процент расхождения между двумя изображениями. В виде кода это будет выглядеть примерно так:

MotionDetector detector = new MotionDetector(
new TwoFramesDifferenceDetector( ),
new MotionAreaHighlighting( ) );


//Обработка очередного кадра
if ( detector != null )
{
float motionLevel = detector.ProcessFrame( image );

if ( motionLevel > motionAlarmLevel )
{
flash = (int) ( 2 * ( 1000 / alarmTimer.Interval ) );
}


if ( detector.MotionProcessingAlgorithm is BlobCountingObjectsProcessing )
{
BlobCountingObjectsProcessing countingDetector = (BlobCountingObjectsProcessing) detector.MotionProcessingAlgorithm;
objectsCountLabel.Text = "Objects: " + countingDetector.ObjectsCount.ToString( );
}
else
{
objectsCountLabel.Text = "";
}

}
}

Вышеприведенный код (не считая инициализацию класса MotionDetector) у меня выполняется при получении очередного кадра с веб-камеры. Получив кадр, я выполняю банальное сравнение (метод ProcessFrame): если значение переменной motionlevel больше motionLevelAlarm (0.015f), то значит, надо бить тревогу! Движение обнаружено. На одном из скришотов хорошо видна работа демонстрация детектора движений.

Готовность №4

Веб-камеру можно запросто приспособить для распознавания лиц и создания продвинутого способа лог-она в систему? Если переварив весь этот материал, ты думаешь, что это сложно, то ты ошибаешься! В конце марта на сайте http://codeplex.com (хостинг для OpenSource проектов от MS) появился пример (а затем и ссылка на статью), демонстрирующий реализацию программы для распознавания лиц с использованием веб-камеры. Сам пример основан на использовании новых возможностей .NET и SilverLight. Разобрать этот пример в рамках журнальной статьи нереально, так как автор исходника постарался и сделал все максимально шикарно. Тут тебе и алгоритмы для работы с изображениями (фильтр размытия, уменьшения шума, попиксельное сравнение, растяжка и т.д.) и демонстрация новинок SilverLight и много чего еще. Одним словом, must use! Ссылку на проект и статью ищи ниже.

Конец фильма

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

Вспомни статью про бэкап Skype-бесед. В ней я говорил, что времена клавиатурных шпионов уже прошли. Сейчас особенно актуальны аудио и видеоданные. Если учесть, что сегодня веб-камера – обязательный атрибут любого ноутбука, то нетрудно представить, сколько интересного видео ты сможешь заснять, подсунув жертве "полезную программку"… Однако я тебе этого не говорил :). Удачи в программировании, а будут вопросы – пиши.

WWW

http://blogs.msdn.com/ – Русская версия статьи "Silverlight 4 real-time Face Detection" (распознавание лиц в реальном времени при помощи SilverLight).

http://facelight.codeplex.com/ – здесь хостится проект "Facelight", позволяющий распознавать лица в реальном времени. Если ты собрался закодить серьезную софтину для определения лиц или логона в систему, то посмотреть на этот проект просто обязан.

http://www.aforgenet.com/framework/ – тут ты найдешь AForge .NET – отличный и простой в использовании фреймворк для работы с видео, изображениями и т.д.

http://vr-online.ru – все исходники примеров, а также кучу дополнительной информации ты можешь слить с сайта проекта VR-Online.

веб-камера программирование


Статьи по теме
Следующие статьи
Предыдущие статьи

comments powered by Disqus

UserГость
10.08.2010 16:03:38
Ответить
Автор:Игорь Антонов - BRAVO. Жар почуствулся.
UserГость
11.08.2010 10:26:06
Ответить
Нет ли подобных проектов, библиотек, фреймворков или примеров но на Java?

Был бы очень признателен.
UserГость
11.08.2010 11:36:51
Ответить
Нах себя мучать?
Есть библиотека OpenCV и все что тут расписывалось на десяти листах делается в три строки кода

cvCapture *capture = cvCaptureFromCAM(0);
IplImage* frame = cvQueryFrame(capture);
cvShowImage("wnd1", frame);
UserГость
11.08.2010 11:38:35
Ответить
И кто томуже подглядывать не получиться... чаще всего горит лампочка при включении вебки... сначала подумай как на уровне ядра можно ее выключить) ыы)
UserГость
11.08.2010 17:12:40
Ответить
У кого есть готовый вариант такой софтины?Готов купить.
З Ы
Софт будет приобретен только после демонстрации по тимке и 100%
UserГость
11.08.2010 17:40:54
Ответить
Если поработать то можно сделать готовый вариант
UserГость
11.08.2010 23:27:25
Ответить
Ну так не вопрос.Готовый купил бы.
UserГость
12.08.2010 12:27:26
Ответить
очень любопытно :)
UserГость
13.08.2010 11:06:47
Ответить
Не знаю принято ли тут оставлять ссылки на другие сайты, но все же тут сегодня прочел про пакет удаленного администрирования http://www.ixbt.com/soft/remote-manipulator.shtml , наткнулся в описании на такую функцию как "Удаленная веб-камера", мне кажется хоть и вылазит какая то поибота там с оповещение, но все же это реальная альтернатива мудроебству для обычных пользователей хотящих понаблюдать по вебкамере за кем нибудь из знакомых, ну или не знакомых, как повезет.
UserГость
14.08.2010 8:22:40
Ответить
Зачем покупать этот софт, если можно написать его самому? Учите языки программирования, товарищи!
UserГость
16.08.2010 16:26:46
Ответить
Все бы хорошо....но с чего лучше начать ?) Какой язык программирования выбрать?
З.Ы. В школе мучали паскалем...как-то он мне не по душе был=)
UserГость
20.09.2010 18:04:45
Ответить
а сорцы библиотеки не дадите?
UserГость
15.10.2010 19:18:03
Ответить
Языки программирования не мое. Кто готов написать шпион? Готов купить! Жду ваши предложения на мыло angtim@yandex.ru
UserГость
09.12.2010 16:42:06
Ответить
)))
UserГость
19.01.2011 22:43:16
Ответить
DirectShow есть... а это - мозгоимейство....
UserГость
11.03.2011 20:57:20
Ответить
Я немогу подключить модуль
using WebCamLibrary;
В Чем проблема? что не так?
UserГость
22.03.2011 22:36:32
Ответить
Напишу любой сложности вирус,программу,сайт,занимаюсь программированием с 5 л,недорого
UserГость
09.10.2011 15:43:25
Ответить
quote:

Напишу любой сложности вирус,программу,сайт,занимаюсь программированием с 5 л,недорого


Очень нуждаюсь в подобной услуге !
roman_shelin@mail.ru
UserГость
23.07.2011 11:18:34
Ответить
Программисты C++ смотрят на автора как на %s
UserГость
29.08.2011 1:27:27
Ответить
Мда, де тут хацкерство то??? Честно говоря не ожидал тут подобное встретить) Сама мысль шпиона на шарпе навоять чего только стоит))
UserГость
14.09.2011 15:17:54
Ответить
quote:

Мда, де тут хацкерство то??? Честно говоря не ожидал тут подобное встретить) Сама мысль шпиона на шарпе навоять чего только стоит))

поддерживаю.
UserDjum
14.09.2011 17:13:24
Ответить
Да, удалённая слежка не помешала бы, более актуальнее чем скрытность думаю нет, представте такую прогу которая зарылась глубоко в процессах)
UserГость
11.09.2011 1:44:24
Ответить
Спасибо за статью, очень хорошо!
Кстати у меня в нет буке нет индикатора активности вебки Оо!
UserГость
11.09.2011 1:58:20
Ответить
Жаль что исходник "потерялся")
UserГость
14.09.2011 15:16:34
Ответить
а где собсно вариант с удаленной слежкой?
UserГость
01.12.2011 11:58:09
Ответить
давайте попробуем
UserГость
09.12.2011 12:16:49
Ответить
так есть же простое решение тайвивер с этим всем правляется необходимо лишь настроить неконтролируемый доступ
UserГость
29.12.2011 22:51:37
Ответить
ха
UserГость
28.02.2012 10:57:01
Ответить
даем на обнал карты

атм карты для обналичивания
http://xakepc.in/
UserГость
28.02.2012 23:43:47
Ответить
Здравствуйте всем!!! готов приобрести программу mrgospodin@mail.ru очень нужна!!!
Userвуда
10.03.2012 15:07:10
Ответить
написал на делфи. могу дать саму прогу или исходники.
работает скрытно.
цена вопроса 5 wmz писать в асю 635108220
UserГость
13.03.2013 23:04:18
Ответить
Готов приобрести программу, если она действительно рабочая и функционирует скрытно. saimovt@mail.ru
UserГость
19.03.2012 3:56:20
Ответить
нюхайте все хуй..прога существует уже около года
UserГость
16.05.2012 10:28:16
Ответить
у меня есть простая камера можно ли сделать ее как веб? помогите!!!!! вот мой мир kot_maloi@mail.ru туда пишете спасибо заранее!
UserГость
24.05.2012 17:19:59
Ответить
ниииихуя себе у вас мозгииии,,,,,,,,,,,,, научитееее ее (((=D
UserГость
22.06.2012 22:38:14
Ответить
5+
UserГость
13.12.2012 21:16:11
Ответить
куплю прогу если покажыте через тимвивер!что как куда и как работает!!
UserГость
13.12.2012 21:17:07
Ответить
куплю пишыте на choper01.rambler.ru
UserГость
26.12.2012 11:16:55
Ответить
Как. Стать. Хакером
UserГость
06.01.2013 0:57:45
Ответить
ТОВАРИСССЧ! На дворе 13й год. Неужто с момента показа этого кина ничего не возникло в природе? Из всей этой истории имею лишь IP объекта (и то, сомневаюсь, что это он не провайдерский), голые заявления некоторых о давно существующих прогах и кучу дурацких ссылок на якобы хаки webcam.
Дай ты нам, простым обывателям, побывать на "обратной стороне", проведи по своим электронным тропам к двери с web глазом ;)
Готов терпеливо воспринимать короткие понятные инструкции (видео или текст). Пора! Благодарю!
UserГость
27.03.2013 20:43:30
Ответить
пошло
UserГость
29.09.2013 20:16:11
Ответить
Есть неплохая библиотека TVideoGrabber и для дельфи и для разных типов Си и для Васи.
UserГость
07.10.2013 6:07:08
Ответить
заклеил вебку*
UserГость
14.02.2014 11:32:17
Ответить
http://freehacks.ru/ - Все хакерские направления в одном месте!

Читай «Хакер» на планшетах!

Свежие материалы каждый месяц в удобном для тебя формате:



Линк рекламодателям

Читай новый номер
В продаже с 2 апреля
  • Лучшие инструменты для создания прототипа твоего первого приложения
  • Разбираемся с главными библиотеками для построения графического интерфейса в мире Java
  • Ретроспектива виртуализации в *nix
  • Тестируем фреймворк для мониторинга Sensu
  • Подбираем утилиты для бэкапа БД
  • Подборка свежих гаджетов для пентестеров
  • Физическая безопасность в ИТ: как ненадежный замок может стать самым слабым местом в системе



Социальные сети


Keywords: zPOSTz zHOMEz, zSOFTz, zHOWz, zINFOz, zYANDEXz z52926z
Для Авторов: edit Lock delete Lock

Автор: Игорь Антонов
web site
Дата: 10.08.2010 15:41:24©