如何通过WinAPI使用没有连字的字体?

时间:2015-06-18 20:49:47

标签: c++ winapi

我尝试使用自定义输出创建简单的应用程序。我使用CreateFont函数来加载具有特定选项的等宽字体。结果,文本被绘制(DrawText)与连字符如fi。

如何禁用此功能?

我很抱歉我的英语不好

创建字体

normal = CreateFont(char_height, 0, 0, 0, FW_NORMAL, 0, 0, 0,
                    RUSSIAN_CHARSET, OUT_DEFAULT_PRECIS, 
                    CLIP_DEFAULT_PRECIS, 5, 
                    FIXED_PITCH | FF_DONTCARE, "Menlo");
bold = CreateFont(char_height, 0, 0, 0, FW_BOLD, 0, 0, 0,
                  RUSSIAN_CHARSET, OUT_DEFAULT_PRECIS, 
                  CLIP_DEFAULT_PRECIS, 5, 
                  FIXED_PITCH | FF_DONTCARE, "Menlo");

打印

void Console::Print(string text, int color, int weight) {
    RECT r;
    r.top    = padding + temp_y * (char_height + space_y);
    r.bottom = r.top   + char_height;
    r.left   = padding + temp_x * char_width;
    r.right  = padding + length();
    HDC hdc = GetDC(hwnd);
    SetTextColor(hdc, color);
    SetBkColor(hdc, RGB(43, 48, 59));
    SelectObject(GetDC(hwnd), weight == WEIGHT_NORMAL ? normal : bold);
    DrawText(GetDC(hwnd), text.c_str(), text.length(), &r, DT_LEFT);

    temp_x += text.length();
}

当前输出(注意单词"离线" ):

Current output

期望的输出:

Desired output

1 个答案:

答案 0 :(得分:0)

根本问题似乎是你的字体坏了。我不确定为什么等宽字体(固定音高)字体会有连字信息。

解决方案1 ​​

ExtTextOutWETO_IGNORELANGUAGE一起使用。您可能会丢失其他功能(例如bi-di,数字替换,复杂脚本的整形),但它似乎可以防止字距调整和结扎,因此它可能适合您的目的。

::ExtTextOutW(hdc, x, y, ETO_IGNORELANGUAGE, &rc, msg.c_str(), msg.size(), nullptr);

我测试了字体Gabriola,因为我没有Menlo或Meslo。 Gabriola是一种可变间距字体,但它具有独特的连线,使其易于发现,尤其是fi

解决方案2

第二种方法具有相同的缺点,即在循环中使用TextOut逐字符绘制字符串。这有点棘手,因为您必须担心Unicode代理对,剪切和更新当前位置。

const auto old_alignment = ::SetTextAlign(hdc, TA_UPDATECP);
const auto old_mode = ::SetBkMode(hdc, TRANSPARENT);
::MoveToEx(hdc, x, y, nullptr);
// Loop simplified for demo.  This doesn't handle Unicode surrogate pairs.
for (auto ch : msg) {
    ::TextOutW(hdc, 0, 0, &ch, 1);
}
::SetBkMode(hdc, old_mode);
::SetTextAlign(hdc, old_alignment);

这产生了与第一种解决方案相同的结果。

非解

请注意,我之前使用GetCharacterPlacement而不使用GCP_LIGATE标志后跟ExtTextOut使用ETO_GLYPH_INDEX的想法不起作用。即使没有GCP_LIGATE标志,从GetCharacterPlacement返回的字形仍包含连字。

// DOES NOT WORK
GCP_RESULTSW results = {sizeof(results)};
WCHAR modified[64] = L"";  // FIXED BUFFER LENGTHS JUST FOR TESTING
results.lpOutString = modified;
int deltas[64] = {0};
results.lpDx = deltas;
WCHAR glyphs[64] = L"";
glyphs[0] = 1;
results.lpGlyphs = glyphs;
results.nGlyphs = ARRAYSIZE(glyphs);

const DWORD flags = GCP_REORDER;  // but not GCP_LIGATE or GCP_USEKERNING
::GetCharacterPlacementW(hdc, msg.c_str(), msg.size(), 0, &results, flags);
::ExtTextOutW(hdc, x, y, ETO_GLYPH_INDEX, &rc, glyphs, results.nGlyphs, deltas);