我正在尝试使用C / C ++代码和Win API函数绘制线条。由于特定的原因,我需要有1个像素的线宽,但我在屏幕上得到的是完全不同的。根据笔/画笔设置的方式,它可以是2或3像素宽的线,而不是1。 仅仅为了测试,我试图绘制3像素线,结果得到5像素。 与其他类型的绘图基元相同的故事:圆角矩形,矩形,椭圆等。 有没有办法在绘制的线条周围禁用这种阴影边界? 绘图功能的代码示例如下。
void DisplayDrawings (HDC hDC, HWND hWnd)
{
LOGBRUSH stLogBrush;
RECT rRect = {0,0,0,0};
HBITMAP hBitmap = NULL;
HBRUSH hBrush = NULL;
HGDIOBJ hOldBrush = NULL;
HGDIOBJ hOldPen = NULL;
HPEN hPen = NULL;
BOOL bB_Result = FALSE;
UINT bBrushByBits [8];
int iRightPos = 0;
int iLeftPos = 30;
int iLineY = 30;
int iDrawStep = 10;
int i = 0;
#ifndef RGBA
#define RGBA(r,g,b,a) ((COLORREF)( (((DWORD)(BYTE)(a))<<24) | RGB(r,g,b) ))
#endif
SetBkMode (hDC, OPAQUE);
bB_Result = GetClientRect(hWnd, &rRect);
if (bB_Result == FALSE)
{
return;
}
iRightPos = rRect.right - 30 - 1;
// Line 1. Draw 1 pixel Line using standard brush/pen.
hPen = CreatePen (PS_SOLID,1, RGB(255, 0,0));
hOldPen = SelectObject (hDC, hPen);
if (hOldPen == NULL)
{
return;
}
MoveToEx (hDC, iLeftPos,iLineY,NULL);
LineTo (hDC,iRightPos,iLineY);
SelectObject (hDC, hOldPen);
DeleteObject (hPen);
iLineY += iDrawStep;
// Result - line with 2 pixels width
// Line 2. Draw 1 pixel Line using standard brush/pen, but setting pen width to "0". CreatePen API says: "If nWidth is zero, the pen is a single pixel wide, regardless of the current transformation"
hPen = CreatePen (PS_SOLID,0, RGB(255, 0,0));
hOldPen = SelectObject (hDC, hPen);
if (hOldPen == NULL)
{
return;
}
MoveToEx (hDC, iLeftPos,iLineY,NULL);
LineTo (hDC,iRightPos,iLineY);
SelectObject (hDC, hOldPen);
DeleteObject (hPen);
iLineY += iDrawStep;
// Result - line with 3 pixels width
// Line 3. Draw 1 pixel as above, but set current brush to NULL
hOldBrush = SelectObject (hDC, GetStockObject(NULL_BRUSH));
if (hOldBrush == NULL)
{
return;
}
hPen = CreatePen (PS_SOLID,0, RGB(255, 0,0));
hOldPen = SelectObject (hDC, hPen);
if (hOldPen == NULL)
{
return;
}
MoveToEx (hDC, iLeftPos,iLineY,NULL);
LineTo (hDC,iRightPos,iLineY);
SelectObject (hDC, hOldPen);
DeleteObject (hPen);
SelectObject (hDC, hOldBrush);
iLineY += iDrawStep;
// Result - line with 2 pixels width
// Line 4. Draw 3 pixel Line using standard brush/pen
hPen = CreatePen (PS_SOLID,3, RGB(255, 0,0));
hOldPen = SelectObject (hDC, hPen);
if (hOldPen == NULL)
{
return;
}
MoveToEx (hDC, iLeftPos,iLineY,NULL);
LineTo (hDC,iRightPos,iLineY);
SelectObject (hDC, hOldPen);
DeleteObject (hPen);
iLineY += iDrawStep;
// Result - line with 5 pixels width
// Line 5. Draw 3 pixel as above, but set current brush to NULL
hOldBrush = SelectObject (hDC, GetStockObject(NULL_BRUSH));
if (hOldBrush == NULL)
{
return;
}
hPen = CreatePen (PS_SOLID,3, RGB(255, 0,0));
hOldPen = SelectObject (hDC, hPen);
if (hOldPen == NULL)
{
return;
}
MoveToEx (hDC, iLeftPos,iLineY,NULL);
LineTo (hDC,iRightPos,iLineY);
SelectObject (hDC, hOldPen);
DeleteObject (hPen);
SelectObject (hDC, hOldBrush);
iLineY += iDrawStep;
// Result - line with 5 pixels width
// Line 6. Draw 1 pixel line by creating own pen with monochrome 8x8 brush. Result is 3
int k = 0;
for(i = 0; i < 8; i++)
{
bBrushByBits [i] = 0x00;
}
hBitmap = CreateBitmap(8, 8, 1, 1, (LPBYTE)bBrushByBits);
if (hBitmap == NULL)
{
return;
}
stLogBrush.lbColor = RGBA(255,0,0,0);
stLogBrush.lbHatch = (ULONG_PTR) hBitmap;
stLogBrush.lbStyle = BS_PATTERN;
hPen = ExtCreatePen (PS_GEOMETRIC, 3, &stLogBrush, 0, NULL);
hOldPen = SelectObject (hDC, hPen);
if (hOldPen == NULL)
{
return;
}
MoveToEx (hDC, iLeftPos,iLineY,NULL);
LineTo (hDC,iRightPos,iLineY);
SelectObject (hDC, hOldPen);
DeleteObject (hPen);
DeleteObject (hBitmap);
// Result - line with 2 pixels width
}
提前致谢
答案 0 :(得分:0)
tl;博士:DPI意识。
Windows会为高DPI显示器缩放图形,除非您明确告诉它您的程序了解如何使用实际的DPI。
您可以在应用程序的早期使用清单或系统调用执行此操作。 API的原始版本是SetProcessDPIAware。较新的版本是SetProcessDPIAwareness,它可以让你告诉Windows,即使在监视器具有不同DPI的多监视器系统上你也知道你在做什么。