当前位置:C++技术网 > 精选软件 > win32移动剪切区域OffsetRgn实现字符串的颜色跑马灯效果

win32移动剪切区域OffsetRgn实现字符串的颜色跑马灯效果

更新时间:2016-01-26 23:48:05浏览次数:1+次

    我们先看看我们要实现的效果图:

红色在“+技术网”四个字符上

【红色在“+技术网”四个字符上】

【红色在“ttp://ww”字符上】

【红色在“ttp://ww”字符上】
    这个效果就是让一团颜色覆盖在几个字符上,然后这个颜色还可以滚动,就实现了颜色跑马灯效果。你可以对滚动进行控制,可以让这块红色从左到右,然后再滚动到左边,或者不停地从左到右。不过我们这里只提供基本的滚动的原理讲解。其他的控制滚动的逻辑,就很简单了,你可以自己发挥。
    实际上,颜色跑马灯的颜色滚动原理,在文章《Win32使用剪切区域实现酷狗动态歌词特效:动态歌词颜色滚动覆盖效果》里已经详细的说明了。本文就说一下移动颜色跑马灯效果的原理。
    我们只需要在动态歌词特效里,将合并剪切区域换成移动剪切区域,就实现了跑马灯。其他的所有代码都不变。然而实现的效果却完全不同。
    移动剪切区域函数为OffsetRgn,第一个参数为区域句柄,第二个参数为X轴上的移动距离,大于零向右移动,小于零向左移动。第三个参数就是Y轴上的移动距离,大于零向下移动,小于零则向上移动。
    移动区域后,只是区域的范围值改变了。如果你要让剪切区域移动,则要将移动后的区域设置为当前的剪切区域,这样就达到了移动剪切区域的效果。所以,在后面要调用:
SelectObject(hdc,hRgn);
    这样就将区域选为了剪切区域。我们是完全覆盖的输出字符串,区域移动后,也就只有当前的剪切区域部分能够输出红色的字符串,也就表现出了红色覆盖在当前蓝色字符串上。
    你要实现不同的跑马灯效果,就可以用不同的方式移动当前剪切区域即可。
下面是完整代码:
#include "windows.h"
#include <tchar.h>
TCHAR txt[]=_T("C++技术网http://www.cjjjs.com,提供免费配套的C语言、C++语言、Win32编程、Linux编程、经典书籍下载、零基础编程课程、项目开发经验、学习经验、工作经验、数据结构、算法思维、程序员学习发展等一站式的服务。");
HRGN hRgn;
// - 项目是Unicode字符集
LRESULT CALLBACK WinProc(HWND hwnd,UINT message,WPARAM wParam,LPARAM lParam)
{
    HDC hdc;
    PAINTSTRUCT ps;
    static int i=0;
    static LOGFONT logFont;
    static HFONT hFont;
    static RECT rectClient;
    switch (message)
    {
    case WM_TIMER:
        InvalidateRect(hwnd,NULL,FALSE);
        return 0;
    case WM_CREATE:
        SetTimer(hwnd,1,100,NULL);
        hRgn = CreateRectRgn(50,50,200,100);
        memset(&logFont,0,sizeof(logFont));
        logFont.lfHeight=50;
        logFont.lfWeight=200;
        hFont = CreateFontIndirect(&logFont);
        GetClientRect(hwnd,&rectClient);
            return 0;
        case WM_PAINT:
            {
                hdc = BeginPaint(hwnd,&ps);
                //形成新的区域
                OffsetRgn(hRgn,10,0);//移动区域

                //输出底层的蓝色字符串
                SelectObject(hdc,hFont);
                SetTextColor(hdc,RGB(0,0,255));
                TextOut(hdc,50,50,txt,lstrlen(txt));
                //设置剪切区域
                SelectObject(hdc,hRgn);//剪切区开始生效
                //在相同的位置输出红色字覆盖蓝色的字
                SetTextColor(hdc,RGB(255,0,0));
                TextOut(hdc,50,50,txt,lstrlen(txt));              
                EndPaint(hwnd,&ps);
            }
            return 0;
        case WM_DESTROY:
            DeleteObject(hRgn);//删除区域
            PostQuitMessage(0);
            return 0;
        default:
        break;//跳出到默认处理
    }
    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(WHITE_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|CS_DBLCLKS;

    if(!RegisterClass(&wndClass))return 0;
    HWND hwnd = CreateWindow(ClassName,title1,WS_OVERLAPPEDWINDOW,10,100,800,400,NULL,NULL,hInstance,NULL);
    ShowWindow(hwnd,SW_SHOWNORMAL);

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