Проблемы с использованием Gdiplus::Graphics::DrawText для рисования белого текста на черном фоне

Я пытаюсь найти способ использовать метод Gdiplus::Graphics.DrawString для создания растрового изображения с белым текстом на черном фоне.

(Почему? Потому что мне нужно иметь возможность вращать текст, а объект Gdiplus::Graphics имеет для этого доступный RotateTransform.)

Я написал пример консольного приложения, которое демонстрирует, что я вижу, когда пытаюсь нарисовать белый текст на черном прямоугольнике. (Код вставлен ниже.)

У кого-нибудь есть идеи, что я делаю неправильно? Заранее благодарю за любую помощь.

////////////////////////

// DrawTextTest.cpp : Defines the entry point for the console application.
//
#include "stdafx.h"
#pragma comment (lib,"Gdiplus.lib")
#include <Windows.h>
#include <GdiPlusEnums.h>
#include <GdiPlusTypes.h>
#include <GdiPlus.h>
#include <iostream>
#include <string>

using namespace Gdiplus;
using namespace std;

bool RunTheTest(string);
bool SaveBitmapToFile(HDC, HBITMAP, BITMAPINFO&, string);
bool WriteBitmapFile(BYTE*, BITMAPINFOHEADER&, LPCTSTR);
void ShowError();

#define BUFFER_SIZE 1024

int _tmain(int argc, _TCHAR* argv[])
{
    bool bRet(false);
    char szBuff[BUFFER_SIZE];
    size_t retVal;
    string sFileName, sUserInput;
    ULONG_PTR gdiplusToken;
    GdiplusStartupInput gdiplusStartupInput;

    if (argc > 1)
    {
        for (int nArg = 1; nArg < argc; nArg++)
        {
            wcstombs_s(&retVal, szBuff, (size_t)BUFFER_SIZE,  argv[nArg], (size_t)BUFFER_SIZE );
            if (nArg > 1)
                sFileName += " ";
            sFileName += szBuff;
        }
    }
    else
    {
        cout << "Input the full path filename (do not enclose in quotes)" << endl;
        cin >> sUserInput;
        sFileName += sUserInput;
    }
    Status s = GdiplusStartup(&gdiplusToken, &gdiplusStartupInput, NULL);
    bRet = RunTheTest(sFileName);
    GdiplusShutdown(gdiplusToken);
    if (bRet)
        return 0;
    return 1;
}

bool RunTheTest(string sfileName)
{
    BITMAPINFO bitmapInfo;
    bool bRet(false);
    Brush* pBlackBrush = new SolidBrush(Gdiplus::Color::Black);
    Brush* pWhiteBrush = new SolidBrush(Gdiplus::Color::White);
    double inchesPerMeter(39.3700787);
    float fontSize(30);
    Font* pFont(NULL);
    FontFamily* pFontFamily(NULL);
    FontStyle fs(FontStyleRegular);
    HBITMAP hMemoryBitmap(NULL);
    HDC hMemoryDC(NULL);
    HGDIOBJ hDefaultBitmap(NULL);
    int pelsPerMeter(0), resolution(240);
    int oldBkMode(0);
    LPTSTR pFileName(NULL);
    PointF origin(0, 0);
    SIZE sizeRect;
    Status s;

    // Initialize a memory device context compatible with the screen
    hMemoryDC = CreateCompatibleDC(NULL);

    // Prepare some values for creating a memory bitmap
    pelsPerMeter = (int)(resolution * inchesPerMeter);
    sizeRect.cx = sizeRect.cy = 400;

    // Create the memory bitmap
    bitmapInfo.bmiHeader.biBitCount = 24;
    bitmapInfo.bmiHeader.biCompression = BI_RGB;
    bitmapInfo.bmiHeader.biPlanes = 1;
    bitmapInfo.bmiHeader.biXPelsPerMeter = pelsPerMeter;
    bitmapInfo.bmiHeader.biYPelsPerMeter = pelsPerMeter;
    bitmapInfo.bmiHeader.biSize = sizeof(bitmapInfo.bmiHeader);
    bitmapInfo.bmiHeader.biWidth = sizeRect.cx;
    bitmapInfo.bmiHeader.biHeight = sizeRect.cy;
    hMemoryBitmap = CreateCompatibleBitmap(hMemoryDC, sizeRect.cx, sizeRect.cy);
    hDefaultBitmap = SelectObject(hMemoryDC, hMemoryBitmap);

    // Draw a white rectangle on the bitmap
    SelectObject(hMemoryDC, GetStockObject(WHITE_BRUSH));
    Rectangle(hMemoryDC, 0, 0, sizeRect.cx, sizeRect.cy);
    SelectObject(hMemoryDC, GetStockObject(NULL_BRUSH));

    // Set bitmap background mode to transparent mode
    oldBkMode = SetBkMode(hMemoryDC, TRANSPARENT);

    // Get a Graphics object from the memory device context
    Graphics graphics(hMemoryDC);

    // draw a black rectangle on the bitmap
    s = graphics.FillRectangle(pBlackBrush, 0, 0, 400, 400);
    if (s != Ok)
    {
        cout << "FillRectangle failed" << endl;
        return false;
    }

    // draw white text on the black rectangle using the font we created
    s = graphics.SetTextRenderingHint(TextRenderingHintAntiAlias);
    if (s != Ok)
    {
        cout << "SetTextRenderingHint failed" << endl;
        return false;
    }

    // Create a font object and draw the text on the bitmap
    pFontFamily = new FontFamily(L"Arial");
    if (pFontFamily == NULL)
    {
        cout << "new FontFamily failed" << endl;
        return false;
    }
    pFont = new Font(pFontFamily, fontSize, fs);
    if (pFont == NULL)
    {
        cout << "new Font failed" << endl;
        if (pFontFamily != NULL)
            delete pFontFamily;
        return false;
    }

    s = graphics.DrawString(L"TEST STRING", 11, pFont, origin, pWhiteBrush);
    if (s == Ok)
    {   // Save the bitmap to a file
        bRet = SaveBitmapToFile(hMemoryDC, hMemoryBitmap, bitmapInfo, sfileName);
    }

    // Clean up
    SetBkMode(hMemoryDC, oldBkMode);
    if (pFont != NULL)
        delete pFont;
    if (pFontFamily != NULL)
        delete pFontFamily;
    if (hDefaultBitmap != NULL && hMemoryBitmap != NULL)
        SelectObject(hMemoryDC, hDefaultBitmap);
    if (hMemoryBitmap != NULL)
        DeleteObject(hMemoryBitmap);
    if (hMemoryDC != NULL)
        DeleteDC(hMemoryDC);
    return bRet;
}

bool SaveBitmapToFile(HDC hMemoryDC, HBITMAP hMemoryBitmap, BITMAPINFO& bitmapInfo, string sFileName)
{
    bool bRet(false);
    DWORD dwBmpSize(0);
    BYTE* pBytes(NULL);
    ULONG ulcb(0);
    WCHAR wchBuff[BUFFER_SIZE];
    PSTR pTemp = (PSTR)wchBuff;

    for (unsigned int nChar = 0; nChar < sFileName.length(); nChar++)
    {
        *pTemp++ = sFileName[nChar];
        *pTemp++ = 0;
    }
    *pTemp++ = 0;
    *pTemp = 0;

    dwBmpSize = ((bitmapInfo.bmiHeader.biWidth * bitmapInfo.bmiHeader.biBitCount + 31) / 32) * 4 * bitmapInfo.bmiHeader.biHeight;
    pBytes = new BYTE[dwBmpSize];
    bRet = (GetDIBits(hMemoryDC, hMemoryBitmap, 0, bitmapInfo.bmiHeader.biHeight, &pBytes[0], &bitmapInfo, DIB_RGB_COLORS) != 0);
    if (bRet)
        bRet = WriteBitmapFile(pBytes, bitmapInfo.bmiHeader, wchBuff);
    if (pBytes != NULL)
    {
        delete[] pBytes;
        pBytes = NULL;
    }
    return bRet;
}

bool WriteBitmapFile(BYTE* pBitmapBits, BITMAPINFOHEADER& bmpInfoHeader, LPCTSTR lpszFileName)
{
    BITMAPFILEHEADER bfh = {0};
    bool bRet(false);
    DWORD dwcb2Write(0), dwcbWritten(0);

    // This value should be values of BM letters i.e 0×4D42
    // 0x4D = M 0x42 = B storing in reverse order to match with endian
    bfh.bfType=0x4D42;
    // or...
    // bfh.bfType = ‘B’+(’M’ << 8);
    // <<8 used to shift ‘M’ to end

    // Offset to the RGBQUAD
    bfh.bfOffBits = sizeof(BITMAPINFOHEADER) + sizeof(BITMAPFILEHEADER);
    bfh.bfSize = bfh.bfOffBits + bmpInfoHeader.biSizeImage;

    HANDLE hFile = CreateFile(lpszFileName, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
    if (hFile == INVALID_HANDLE_VALUE)
    {
        cout << "CreateFile failed" << endl;
        ShowError();
        return false;
    }
    dwcb2Write = sizeof(bfh);
    bRet = (WriteFile(hFile, &bfh, dwcb2Write, &dwcbWritten, NULL) != 0);
    if (!bRet || (dwcbWritten != dwcb2Write))
    {
        cout << "WriteFile failed" << endl;
        ShowError();
        CloseHandle(hFile);
        return false;
    }
    dwcb2Write = sizeof(bmpInfoHeader);
    bRet = (WriteFile(hFile, &bmpInfoHeader, dwcb2Write, &dwcbWritten, NULL) != 0);
    if (!bRet || (dwcbWritten != dwcb2Write))
    {
        cout << "WriteFile failed" << endl;
        ShowError();
        CloseHandle(hFile);
        return false;
    }
    dwcb2Write = bmpInfoHeader.biSizeImage;
    bRet = (WriteFile(hFile, pBitmapBits, dwcb2Write, &dwcbWritten, NULL) != 0);
    if (!bRet || (dwcbWritten != dwcb2Write))
    {
        cout << "WriteFile failed" << endl;
        ShowError();
        CloseHandle(hFile);
        return false;
    }
    CloseHandle(hFile);
    return true;
}

void ShowError()
{   // Retrieve the system error message for the last-error code
    char szBuff[BUFFER_SIZE];
    LPTSTR lpMsgBuf;
    size_t retVal;
    DWORD dw = GetLastError(), dwRet(0);

    dwRet = FormatMessage(
        FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS,
        NULL,
        dw,
        MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
        (LPTSTR) &lpMsgBuf,
        0, NULL );

    wcstombs_s(&retVal, szBuff, (size_t)BUFFER_SIZE, lpMsgBuf, (size_t)BUFFER_SIZE );

    // Display the error message
    cout << szBuff << endl;

    LocalFree(lpMsgBuf);
}

person Les Amack    schedule 31.03.2011    source источник


Ответы (2)


Просто сначала нарисуйте прямоугольник с помощью Graphics::FillRectangle(). Получите нужный размер прямоугольника из Graphics::MeasureString().

person Hans Passant    schedule 31.03.2011
comment
Но это то, что я уже делаю, как в примере кода. Я неправильно тебя понимаю? - person Les Amack; 31.03.2011
comment
Есть ли какие-либо значения, которые я устанавливаю в BITMAINFO, которые могут быть причиной этого? Я также отмечаю, что когда я использую белую кисть для рисования текста на белом фоне, текст все равно получается черным. ?? - person Les Amack; 31.03.2011
comment
Эм, хорошо, я вижу вызов FillRectangle. Размер 400x400 кажется произвольным. Опишите, что вы ожидаете от того, что вы на самом деле получаете. - person Hans Passant; 31.03.2011
comment
И используйте класс Bitmap для создания растрового изображения. - person Hans Passant; 31.03.2011
comment
В ПОРЯДКЕ. (Прошу прощения, если я не совсем понятен.) Я ожидаю увидеть черный прямоугольник размером 400x400 и некоторый белый текст поверх него шрифтом Arial. Вместо этого я получаю просто черный прямоугольник. - person Les Amack; 31.03.2011
comment
Если вместо этого я нарисую прямоугольник белым цветом, а затем нарисую на нем белый текст, я получу черный текст на белом. - person Les Amack; 31.03.2011
comment
Что-то не так с растровым изображением. Сделайте фоновую кисть желтой, чтобы текст отображался. Используйте класс Bitmap. - person Hans Passant; 31.03.2011
comment
У меня есть функция, которая создает Gdiplus::Bitmap из приведенного выше. Соответствующие строки: GetDIBits(hMemoryDC, hMemoryBitmap, 0, (UINT)bitmapInfo.bmiHeader.biHeight, pBytes, (BITMAPINFO*)&bitmapInfo, DIB_RGB_COLORS); Bitmap* pBitmap = новый Gdiplus::Bitmap(&bitmapInfo, pBytes); Но куда это меня приведет? Могу ли я затем записать растровое изображение через объект Gdiplus::Bitmap? - person Les Amack; 31.03.2011
comment
Ганс, спасибо за ответ! Я уже должен был поблагодарить тебя. - person Les Amack; 31.03.2011
comment
Используйте Graphics::FromImage() для создания объекта Graphics. Используйте Bitmap::Save(), чтобы сохранить его. - person Hans Passant; 31.03.2011
comment
Ганс, ты попал! Хорошо, я создаю memoryDC, использую CreateCompatibleBitmap для memoryBmp, выбираю memoryBmp в memoryDC, создаю Gdiplus::Bitmap из приведенного выше и создаю Gdiplus::Graphics с помощью Graphics::FromImage(pGdiplusBmp). Я не могу тебя отблагодарить. - person Les Amack; 01.04.2011

К сожалению, вы не описываете результат, но я могу сказать по своему опыту, что иногда бывает сложно смешивать вызовы Win32 с GDI-Plus. У меня противоположный эффект: после рисования заполненных областей с помощью GDI-Plus Win32 DrawText иногда не работает. Поэтому я предлагаю использовать GDIPlus для всех вызовов, а не смешивать вызовы Win32.

GDIPLus - это беспорядок. Его заголовки C++ совершенно неполны и не имеют нескольких полезных вещей.

Я только что нашел эту статью, описывающую некоторые ограничения смешивания GDI и GDI+: https://support.microsoft.com/en-gb/help/311221/info-interoperability-between-gdi-and-gdi

person RED SOFT ADAIR    schedule 31.03.2011
comment
Я вызываю Gdiplus::Gaphics.FillRectangle с помощью черной кисти, а затем вызываю Gdiplus::Gaphics.DrawString с помощью белой кисти. Я ожидаю увидеть белый текст на черном прямоугольнике. Вместо этого я вижу только черный прямоугольник, белый текст не появляется, как ожидалось. Если вместо этого я вызову Gdiplus::Gaphics.FillRectangle с помощью белой кисти, а затем вызову Gdiplus::Gaphics.DrawString с помощью белой кисти, я увижу черный текст на белом прямоугольнике. - person Les Amack; 31.03.2011
comment
Ваш код включает вызов Win32 Rectangle(hMemoryDC, 0, 0, sizeRect.cx, sizeRect.cy); Затем вы вызываете graphics.FillRectangle для того же hMemoryDC. Это может быть проблемой. - person RED SOFT ADAIR; 31.03.2011
comment
Истинный. Хорошо, я изменил код, чтобы использовать только Gdiplus::Graphics, но получил тот же результат. - person Les Amack; 31.03.2011
comment
Вы заменили все вызовы Win32? - person RED SOFT ADAIR; 31.03.2011
comment
Кулак, позвольте поблагодарить вас за ответ! Да, я удалил все вызовы Win32, кроме hMemoryDC = CreateCompatibleDC(NULL); hMemoryBitmap = CreateCompatibleBitmap(hMemoryDC, sizeRect.cx, sizeRect.cy); hDefaultBitmap = SelectObject(hMemoryDC, hMemoryBitmap); Могу ли я каким-то образом опубликовать измененный код? В любом случае, я все еще сталкиваюсь с проблемой. - person Les Amack; 31.03.2011
comment
Теперь работает. Я создаю memoryDC, использую CreateCompatibleBitmap для memoryBmp, выбираю memoryBmp в memoryDC, создаю Gdiplus::Bitmap из приведенного выше и создаю Gdiplus::Graphics с помощью Graphics::FromImage(pGdiplusBmp). Я не могу тебя отблагодарить. - person Les Amack; 01.04.2011
comment
Так была ли проблема в смеси GDI и GDI+? (пожалуйста, поблагодарите голосами). - person RED SOFT ADAIR; 04.04.2011