КГ_6глава (1024115), страница 2
Текст из файла (страница 2)
2. Рисование с помощью графических функций API Windows.
3. Уничтожение, деактивизация контекста соответствующего hdc.
Контекст окна на экране дисплея
Для рисования в окне программы на экране дисплея можно использовать ди способа. Эти способы различаются как по особенностям получения значена hdc, так и по возможностям рисования.
Первый способ основывается на использовании пары функций GetDC ij
Функция GetDC получает hdc для окна, заданного кодом hwnd. Например, для главного окна программы. Использование контекста графического вывода завершается вызовом функции ReieaseDC. Другую функцию для освобождения контекста в этом случае использовать не следует. Такой способ работы с контекстом использован в приведенном выше примере программы. Вообще этот способ можно рекомендовать везде, за исключением рисования во врем» обработки сообщения wmpaint.
Второй способ. Используется исключительно в теле обработчика сообщений
|
Во время обработки сообщения wm_paint функция BeginPaint обязательно должна вызываться первой, a EndPaint — последней.
Когда необходимо обрабатывать сообщения wmpaint? Это сообщение присылается любой программе тогда, когда повреждено изображение клиентской области окна этой программы. Повреждение изображения окна случаются довольно часто, например, при отображении на экране нескольких окон — когда на окно было наложено еще одно окно, а потом после закрытия последнего окна части изображения первого окна уже нет. Если в программе предусмотрена перерисовка изображения рабочей (клиентской) области окна
путем программирования обработчика сообщения wmpaint, to это позволяет в большинстве случаев гарантировать корректное отображение окна.
Приведем пример программы, рисующей согласно второму способу.
Контекст принтера
Рассмотрим, как можно нарисовать (а точнее, напечатать) что-то на принтере. Для этого также сначала необходимо получить значение hdc соответствующего графического устройства. Потом выполняется рисование. А затем контекст освобождается. Общая схема такая же. Пример программы:
Для принтера значение hdc получить несколько сложнее. Сначала необходимо узнать имя принтера— здесь это делается с помощью функции EnumPrinters. Обратите внимание на то, что использована не одна структура, а массив из трех структур pinfo5 — это сделано согласно рекомендациям [20]. Потом создается контекст вывода с помощью функции CreateDC. Печать на принтере выполняется С ПОМОЩЬЮ функций StartDoc, StartPage, EndPage и EndDoc. После вызова функции EndDoc контекст принтера необходимо освободить С ПОМОЩЬЮ функции DeleteDC.
Следует заметить, что приведенный пример предназначен для Windows 95, 98. Для Windows NT функцию DrawstudyExampie необходимо модифицировать — об этом можно узнать в документации Win32 SDK [61].
После печати на бумаге можно заметить отличия результата от изображения на экране. Прежде всего, это касается размера прямоугольников. Для лазерного принтера размеры значительно меньше, а для матричного размеры могут быть и больше. Это объясняется различными разрешающими способностями (dpi) принтеров и экрана дисплея. Кроме того, на матричном принтере рисунок может изменить пропорции— в случае, когда (dpi) по вертикали отличается от (dpi) по горизонтали. При вызове функции Rectangle мы использовали координаты в виде пикселов. Функции API Windows разрешают использовать и другие системы координат.
Контекст метафайла
Рассмотрим еще одну разновидность контекста — контекст метафайла. Зда рисунок создается в виде файла на диске. Известны два формата метафайл! для Windows: WMF и EMF. Формат EMF более совершенный. В качеств примера приведем текст программы, записывающей рисунок в файл *М после выбора пункта меню "Записать Как".
I Запустите программу и выберите меню "Файл \ Записать как". После определения имени файла и записи метафайла на диск в окне должно появиться следующее изображение (рис. 6.2). \
"i
Рис. 6.2. Пример метафайла
Метафайл описывает изображение в виде последовательности вызовов графических функций, использованных при создании изображения (в приведенном примере программы это 10 вызовов функции Rectangle). В метафайл записываются функции вместе со значениями аргументов-координат, границы рисунка и другая информация. Общие параметры рисунка можно узнать, если прочитать заголовок метафайла с помощью функции GetEnhMetaFileHeader.
Для отображения метафайла нужно задать границы вывода. Поскольку метафайл — векторное описание изображения, то это дает возможность при отображении плавно растягивать рисунок, причем намного лучше, чем растровые изображения. В нашей программе метафайл отображен 18 раз в контексте главного окна. Обратите внимание на крестик— здесь использовано зеркальное отображение.
В результате работы этой программы на диске должен появиться файл *.emf. Чтобы проверить содержимое файла, вызывайте любую программу, которая может читать файлы формата EMF — например, Word для Windows. Выполните вставку рисунка *.emf и вы увидите десять прямоугольников. Рисунок! векторный, поэтому его можно свободно растягивать. Границы рисунка определяются согласно диапазону координат всех элементов рисунка — аргументов вызова функции Rectangle.
Необходимо заметить, что нижний прямоугольник этого рисунка обычно выглядит поврежденным (как именно — это еще зависит от версии Word).
Этот прямоугольник рисуется согласно координатам при i = 9:
Rectangle (hdc, 9*30, 9*20, 9*30+70, 9*20+50). Координаты правого нижнего угла (9*30+70, 9*20+50) этого прямоугольника определяют правую и нижнюю границу всего рисунка. Границы рисунка записываются в метафайл в виде координат левого верхнего и правого нижнего углов. Чтобы узнать, какие границы определены для метафайла, можно использовать функцию
1 Эта функция заполняет поля структуры emh, имеющей тип enhmetaheader. Границы рисунка записываются в поля emh. rclBounds, и имеют такие значения:
Обратите внимание на то, что координаты правого нижнего угла имеют значения на единицу меньше, чем должны быть. Это — проявление особенностей функции Rectangle, рисующей правый нижний угол контура как раз на один пиксел ближе к левому верхнему углу. К тому можно привыкнуть, но почему Microsoft Word (версий 6.0 для 95 и 97-й) искажает рисунок (отрезает границы крайнего прямоугольника)? Средствами Word можно расширить границы этого рисунка, но для других файлов *. emf это может привести, например, к повреждению стиля линий. Поэтому попробуем расширить границы рисунка собственноручно, например, так:
Здесь мы сначала рисуем один белый пиксел с помощью функции SetPixel, что никак не изменяет сам рисунок, но координаты его (х,у) = (9*30+70, 9*20+50) = (340, 230) будут учтены при определении границ рисунка. Запустите модифицированную программу, запишите метафайл, и попробуйте загрузить этот метафайл в редактор Word. Рисунок должен выглядеть неповрежденным.
Кроме того, в некоторых рисунках, которые записываются в метафайлы, необходимо учитывать толщину пера. В нашем случае используется перо по
умолчанию, имеющее толщину в один пиксел. Если рисовать толстыми перьями, то это также может привести к ошибкам определения границ рисунка не только для правого нижнего, а и левого верхнего углов. Рассмотрим так пример:
В приведенном примере две функции set Pixel использованы для установления границ рисунка (-1, -1, 341, 231). Если выбросить эти две функции, границы будут (0, 0, 339, 229), что может привести к проблемам с использованием метафайла в программе Word.
Необходимо отметить, что в нашей программе функция API PiayEnhMetaFile корректно отображает этот метафайл, поэтому к ней никаких претензий, i расширение границ с помощью функции setpixel — это попытка исправив ошибки соответствующих прикладных программ.
Если при создании метафайла его имя отсутствует, например] CreateEnhMetaFile (null, null, null, null) , то метафайл не будет записывать^ на диск, поверхность контекста будет существовать только в памяти компькк тера. Такой метафайл также можно использовать, например, отображать в
ДРУГОМ КОНТексте С ПОМОЩЬЮ функции PiayEnhMetaFile.
Контекст памяти
Рассмотрим контекст памяти (memory context), для которого поверхность рисования создается на основе битовой карты (bitmap).
Контекст в памяти создается вызовом функции CreateCompatibieDC. Здесь он создается по образцу контекста окна программы. Но в таком контексте отсутствует поверхность рисования. Эту поверхность создаем по образцу цветового формата поверхности окна экрана вызовом функций CreateCompatibleBitmap И SelectObject. ПОТОМ МОЖНО рисовать В соответствии с hdc. Сначала мы очищаем поверхность (закрашиваем белым цветом), а потом рисуем десять прямоугольников. Потом вызов функции BitBlt, копирующей растровое изображение из контекста памяти (hdc) в контекст окна (hdcwin). После использования контекста памяти его необходимо закрыть и освободить выделенную память, которая была выделена. Здесь это сделано функциями DeleteDC И DeleteObject.
Контекст памяти часто используется для "заэкранного" рисования — например, для анимации. Рассмотрим один из простейших примеров анимации — в цикле рисуется по десять прямоугольников, размеры которых изменяются от 5 до 300. Это мы сделаем на основе текста программы для предыдущих примеров.
Изображение кажется подвижным, но поскольку здесь все рисуется негой средственно в окне программы, то это воспринимается плохо. Сравните с pfc ботой следующего примера, где использован контекст памяти.