当前位置:C++技术网 > 精选软件 > win32如何画粗虚线,自己实现画粗虚线

win32如何画粗虚线,自己实现画粗虚线

更新时间:2015-12-20 18:02:32浏览次数:1+次

    在学习Win32时,你知道,画笔有几种线型,有PS_SOLID实线,PS_DASH虚线等。我要画虚线,可以设置线宽为0或者1,而设置为0,还是会用1来画。系统会自动处理,不会真用0线宽来画,不然不就等于没有画。
    画实线的时候,你设置线宽大于1之后,可以发现,线宽越大,线越粗。这是很自然的事情。然而,你画了虚线之后,发现太小了,想画粗一点。然后你会发现,将线宽设置大于1之后,虚线成了实线,而不是想要的粗虚线。
    此时怎么办?如果你就是要画粗虚线怎么样实现呢?你不能依赖系统提供的函数来实现,你可以自己实现嘛。其实也很简单,就是自己画多条虚线拼成粗虚线就好了。

    下面是效果图:

红色与黄色相间的虚线PS_DASH和蓝色和青色相间的虚点点线PS_DASHDOTDOT

红色与黄色相间的虚线PS_DASH和蓝色和青色相间的虚点点线PS_DASHDOTDOT


    原理很简单,就是画了一条线之后,将起点和终点向下平移一个像素后画一条同样的线,然后再向下平移后画,如此一来,移动多少次,就决定你要画多宽的虚线了哦。
    图中两条线样式不同和颜色不同而已,然后就是线条的起点和终点不一样,仅此而已。第一条线的起点都是从客户区顶角点开始的,然后都往下平移。结束的点都是在客户区水平方向的中点,高度在垂直方向的中点。

    这里要注意的问题就是,我们画的虚线,断开的位置默认使用白色来填充的,所以我们用白色客户区就看到时断续的线,然而在黑色客户区中看到的就是红白和蓝白相间的线条。不必惊讶,就是背景色的原因。默认用白色填充虚线断开的部分的效果图如下:

默认用白色填充虚线断开的部分的效果图

默认用白色填充虚线断开的部分的效果图

    因为这个加粗了的虚线,看起来感觉不直观,我们将线宽设置为1,即不画多条并排组成粗虚线,看到的效果就是下面这样的:

黑色背景下的单像素的虚线效果图

黑色背景下的单像素的虚线效果图

    如果你还是不适应,那就将客户区背景设置为白色,就还给你一个默认看到的效果:

白色背景下的单像素的虚线效果图

白色背景下的单像素的虚线效果图

    因为背景是白色的,默认填充虚线断开的位置也是用白色的,所以就看到的是这样的断开的虚线。这个在黑色背景就看出来了。不过我们在实现时主动修改了背景颜色,就实现了红黄相间的虚线,这个颜色可以自由设置。当然你也可以将虚线背景填充颜色设置为黑色,就可以看到断续的虚线了。效果如下:

黑色背景设置黑色的填充颜色的虚线效果

黑色背景设置黑色的填充颜色的虚线效果

    因为画虚线是很简单的,所以额外讲了这些内容,丰富一些。

    以下是完整代码,代码就不多解释了:


#include "windows.h"
#include <Windowsx.h>//GET_X_LPARAM,GET_Y_LPARAM宏需要这个头文件
#include <tchar.h>
// - 项目是Unicode字符集
LRESULT CALLBACK WinProc(HWND hwnd,UINT message,WPARAM wParam,LPARAM lParam)
{
    HDC hdc;
    PAINTSTRUCT ps;
    TCHAR Info[100]=_T("C++技术网http://www.cjjjs.com");
    RECT rect;
    switch (message)
    {
     case WM_PAINT:
         hdc = BeginPaint(hwnd,&ps);
         GetClientRect(hwnd,&rect);
         SelectObject(hdc,CreatePen(PS_DASH,1,RGB(255,0,0)));//选择使用画笔
         SetBkColor(hdc,RGB(255,255,0));//设置背景颜色
         for (int i=0;i<50;i++)
         {
             MoveToEx(hdc,0,i,NULL);
             LineTo(hdc,rect.right/2,rect.bottom/2+i);
             LineTo(hdc,rect.right,rect.top+i);
         }
         DeleteObject(SelectObject(hdc,GetStockObject(WHITE_PEN)));//删除画笔

         SelectObject(hdc,CreatePen(PS_DASHDOT,1,RGB(0,0,255)));//选择使用画笔
         SetBkColor(hdc,RGB(0,255,255));//设置背景颜色
         for (int i=0;i<50;i++)
         {
             MoveToEx(hdc,0,rect.bottom+i,NULL);
             LineTo(hdc,rect.right/2,rect.bottom/2+i-50);
             LineTo(hdc,rect.right,rect.bottom+i-50);
         }
         DeleteObject(SelectObject(hdc,GetStockObject(WHITE_PEN)));//删除画笔
         EndPaint(hwnd,&ps);
         return 0;
     case WM_DESTROY:
         PostQuitMessage(0);
         return 0;
     default:
         return DefWindowProc(hwnd, message, wParam, lParam);
    }
}
int WINAPI WinMain(HINSTANCE hInstance,HINSTANCE hPrev,LPSTR lpCmd,int iShow)
{
    TCHAR ClassName[] = _T("MyClass");
    TCHAR title1[] = _T("C++技术网http://www.cjjjs.com");
    WNDCLASS wndClass;
    wndClass.cbClsExtra=0;
    wndClass.cbWndExtra=0;
    wndClass.hbrBackground= (HBRUSH)GetStockObject(BLACK_BRUSH);
    wndClass.hCursor=LoadCursor(NULL,IDC_ARROW);
    wndClass.hIcon=LoadIcon(NULL,IDI_APPLICATION);
    wndClass.hInstance = hInstance;
    wndClass.lpfnWndProc = WinProc;
    wndClass.lpszClassName = ClassName;
    wndClass.lpszMenuName=NULL;
    wndClass.style=CS_HREDRAW|CS_VREDRAW;

    if(!RegisterClass(&wndClass))
    {
        return 0;
    }

    HWND hwnd = CreateWindow(ClassName,title1,WS_OVERLAPPEDWINDOW,0,0,440,400,NULL,NULL,hInstance,NULL);
    ShowWindow(hwnd,SW_SHOWNORMAL);

    MSG msg;
    while (GetMessage(&msg,NULL,0,0))
    {
        TranslateMessage(&msg);
        DispatchMessage(&msg);
    }
    return msg.wParam;
}