TCPMP播放器UI的修改方法

简介:  TCPMP播放器UI的修改方法     前段时间对TCPMP程序进行了研究,花了点时间把TCPMP程序的UI修改成了自已想要的样子,现对UI的修改方法简单介绍下:     网上有文章对于TCPMP程序在ARMV4下的编译方法和程序的结构介绍比较详细,但对于inteface方面的修改方法讲得并不多.

 TCPMP播放器UI的修改方法

    前段时间对TCPMP程序进行了研究,花了点时间把TCPMP程序的UI修改成了自已想要的样子,现对UI的修改方法简单介绍下:

    网上有文章对于TCPMP程序在ARMV4下的编译方法和程序的结构介绍比较详细,但对于inteface方面的修改方法讲得并不多.

    修改TCPMP界面基本上有两种方法:

1)     建立自已的工程,把TCPMP下的lib移植到这个工程里,这样界面的修改比较灵活,但是工作量比较大;

2)     直接在TCPMP工程里修改interface,这样工作量相对比较小,但是UI的修改受了定的约束, 没有前种方法灵活.

    本人所修改的UI是按照第二种方法来做的.主要是对win_win32.c文件和inteface.c 文件进行修改.

   首先,我们来看看win_win32.c文件里的Win_Init()函数,这个函数可以看作是TCPMP UI部分的一个入口,在这个函数里注册了两个窗口类,WinClass和DialogClass.另外还调用了其它控件的初始化函数,下面是这个函数的代码.

view plaincopy to clipboardprint?
void Win_Init()  
{  
    HMODULE Module = Context()->LoadModule;  
    if (!Module) Module = GetModuleHandle(NULL);  
    InitCommonControls();  
    WidcommAudio_Init();  
    stprintf_s(WinClassName,TSIZEOF(WinClassName),T("%s_Win"),Context()->ProgramName);  
 
    memset(&WinClass,0,sizeof(WinClass));  
    WinClass.style  = CS_HREDRAW | CS_VREDRAW | CS_DBLCLKS;  
    WinClass.lpfnWndProc    = Proc;  
    WinClass.cbClsExtra     = 0;  
    WinClass.cbWndExtra     = 0;  
    WinClass.hInstance      = Module;  
    WinClass.hIcon  = LoadIcon   (GetModuleHandle(NULL), MAKEINTRESOURCE(WIN_ICON));  
    WinClass.hCursor        = WinCursorArrow();  
    WinClass.hbrBackground  =NULL;  
    WinClass.lpszMenuName   = 0;  
    WinClass.lpszClassName  = WinClassName;  
    RegisterClass(&WinClass);  
#if 1  
    memset(&DialogClass,0,sizeof(DialogClass));  
    DialogClass.style   = CS_HREDRAW | CS_VREDRAW;  
    DialogClass.lpfnWndProc     = DialogProc;  
    DialogClass.cbClsExtra      = 0;  
    DialogClass.cbWndExtra      = 0;  
    DialogClass.hInstance       = Module;  
    DialogClass.hCursor     =  WinCursorArrow();  
#if defined(TARGET_WINCE)  
    DialogClass.hbrBackground   = GetSysColorBrush(COLOR_STATIC);//   
#else             
    DialogClass.hbrBackground   = GetSysColorBrush(COLOR_BTNFACE);  
#endif  
    DialogClass.lpszMenuName    = 0;  
    DialogClass.lpszClassName   = T("DialogBase");  
    RegisterClass(&DialogClass);  
#endif  
      
    memset(&FontCache,0,sizeof(FontCache));  
 
#if defined(TARGET_WINCE)  
    if (Context()->ProgramId >= 3 && !QueryAdvanced(ADVANCED_OLDSHELL))  
    {  
    AygShell = LoadLibrary(T("aygshell.dll"));  
    *(FARPROC*)&FuncSHCreateMenuBar = GetProcAddress(AygShell,T("SHCreateMenuBar"));  
    *(FARPROC*)&FuncSHInitDialog = GetProcAddress(AygShell,T("SHInitDialog"));  
    *(FARPROC*)&FuncSHFullScreen = GetProcAddress(AygShell,T("SHFullScreen"));  
    *(FARPROC*)&FuncSHHandleWMActivate = GetProcAddress(AygShell,MAKEINTRESOURCE(84));  
    *(FARPROC*)&FuncSHHandleWMSettingChange = GetProcAddress(AygShell,MAKEINTRESOURCE(83));  
    *(FARPROC*)&FuncSHSendBackToFocusWindow = GetProcAddress(AygShell,MAKEINTRESOURCE(97));  
    }  
    CoreDLL = LoadLibrary(T("coredll.dll"));  
    *(FARPROC*)&FuncUnregisterFunc1 = GetProcAddress(CoreDLL,T("UnregisterFunc1"));  
    *(FARPROC*)&FuncAllKeys = GetProcAddress(CoreDLL,T("AllKeys"));  
    *(FARPROC*)&FuncSipShowIM = GetProcAddress(CoreDLL,T("SipShowIM"));  
    *(FARPROC*)&FuncSipGetInfo = GetProcAddress(CoreDLL,T("SipGetInfo"));  
#endif  
    NodeRegisterClass(&Win);  
    QueryKey_Init();  
    OpenFile_Init();  
    Interface_Init();  
    PlaylistWin_Init();  
    PlaylistNewWin_Init();    

void Win_Init()
{
    HMODULE Module = Context()->LoadModule;
    if (!Module) Module = GetModuleHandle(NULL);
    InitCommonControls();
    WidcommAudio_Init();
    stprintf_s(WinClassName,TSIZEOF(WinClassName),T("%s_Win"),Context()->ProgramName);

    memset(&WinClass,0,sizeof(WinClass));
    WinClass.style = CS_HREDRAW | CS_VREDRAW | CS_DBLCLKS;
    WinClass.lpfnWndProc = Proc;
    WinClass.cbClsExtra  = 0;
    WinClass.cbWndExtra  = 0;
    WinClass.hInstance  = Module;
    WinClass.hIcon = LoadIcon   (GetModuleHandle(NULL), MAKEINTRESOURCE(WIN_ICON));
    WinClass.hCursor  = WinCursorArrow();
    WinClass.hbrBackground =NULL;
    WinClass.lpszMenuName = 0;
    WinClass.lpszClassName = WinClassName;
    RegisterClass(&WinClass);
#if 1
    memset(&DialogClass,0,sizeof(DialogClass));
    DialogClass.style = CS_HREDRAW | CS_VREDRAW;
    DialogClass.lpfnWndProc  = DialogProc;
    DialogClass.cbClsExtra  = 0;
    DialogClass.cbWndExtra  = 0;
    DialogClass.hInstance  = Module;
    DialogClass.hCursor  =  WinCursorArrow();
#if defined(TARGET_WINCE)
    DialogClass.hbrBackground = GetSysColorBrush(COLOR_STATIC);//
#else   
    DialogClass.hbrBackground = GetSysColorBrush(COLOR_BTNFACE);
#endif
    DialogClass.lpszMenuName = 0;
    DialogClass.lpszClassName = T("DialogBase");
    RegisterClass(&DialogClass);
#endif
 
    memset(&FontCache,0,sizeof(FontCache));

#if defined(TARGET_WINCE)
    if (Context()->ProgramId >= 3 && !QueryAdvanced(ADVANCED_OLDSHELL))
    {
 AygShell = LoadLibrary(T("aygshell.dll"));
 *(FARPROC*)&FuncSHCreateMenuBar = GetProcAddress(AygShell,T("SHCreateMenuBar"));
 *(FARPROC*)&FuncSHInitDialog = GetProcAddress(AygShell,T("SHInitDialog"));
 *(FARPROC*)&FuncSHFullScreen = GetProcAddress(AygShell,T("SHFullScreen"));
 *(FARPROC*)&FuncSHHandleWMActivate = GetProcAddress(AygShell,MAKEINTRESOURCE(84));
 *(FARPROC*)&FuncSHHandleWMSettingChange = GetProcAddress(AygShell,MAKEINTRESOURCE(83));
 *(FARPROC*)&FuncSHSendBackToFocusWindow = GetProcAddress(AygShell,MAKEINTRESOURCE(97));
    }
    CoreDLL = LoadLibrary(T("coredll.dll"));
    *(FARPROC*)&FuncUnregisterFunc1 = GetProcAddress(CoreDLL,T("UnregisterFunc1"));
    *(FARPROC*)&FuncAllKeys = GetProcAddress(CoreDLL,T("AllKeys"));
    *(FARPROC*)&FuncSipShowIM = GetProcAddress(CoreDLL,T("SipShowIM"));
    *(FARPROC*)&FuncSipGetInfo = GetProcAddress(CoreDLL,T("SipGetInfo"));
#endif
    NodeRegisterClass(&Win);
    QueryKey_Init();
    OpenFile_Init();
    Interface_Init();
    PlaylistWin_Init();
    PlaylistNewWin_Init(); 
}

 

    在此函数中我们需要注意    WinClass.lpfnWndProc= Proc;   Proc是消息处理函数名.实际上所有的消息处理都是在static LRESULT CALLBACK Proc(HWND Wnd, UINT Msg, WPARAM wParam, LPARAM lParam)函数里完成的.

下面对此函数中比较重点的消息作下说明:

    WM_CREATE消息里主要是建立播放的窗口;

   WM_PAINT消息里可以贴上UI的背景图片,这样使UI看上去比较美观.贴图部分程序如下:

 view plaincopy to clipboardprint?
case WM_PAINT:  
    {  
    hdc = BeginPaint(Wnd,&Paint);  
 
    MainBkGnd = LoadBitmap(p->Module,MAKEINTRESOURCE(IDB_MAIN_BKG));  
    MemDC = CreateCompatibleDC(hdc);  
    bmpOld = (HBITMAP)SelectObject(MemDC,MainBkGnd);  
    BitBlt(hdc,0,0,LCD_XSIZE,LCD_YSIZE,MemDC,0,0,SRCCOPY);  
    SelectObject(MemDC,bmpOld);  
    DeleteObject(bmpOld);     
    DeleteDC(MemDC);  
 
    EndPaint(Wnd,&Paint);  
   }  
   break; 
case WM_PAINT:
    {
    hdc = BeginPaint(Wnd,&Paint);

    MainBkGnd = LoadBitmap(p->Module,MAKEINTRESOURCE(IDB_MAIN_BKG));
    MemDC = CreateCompatibleDC(hdc);
    bmpOld = (HBITMAP)SelectObject(MemDC,MainBkGnd);
    BitBlt(hdc,0,0,LCD_XSIZE,LCD_YSIZE,MemDC,0,0,SRCCOPY);
    SelectObject(MemDC,bmpOld);
    DeleteObject(bmpOld); 
    DeleteDC(MemDC);

    EndPaint(Wnd,&Paint);
   }
   break;
 

         WM_COMMAND消息是所有菜单 ,按钮等点击后处理的入口点,具体的实现在interface.c里的static int Command(intface* p,int Cmd)函数里来做处理。如播放,暂停,前一首,下一首等.还有其它的一些消息处理在此不做介绍了.

        Interface.c的程序结构和win_win32.c  基本差不多,其中最主要的也是static bool_t Proc(intface* p, int Msg, uint32_t wParam, uint32_t lParam, int* Result)函数对消息的处理.在此函数的WM_CREATE消息里可以建立起各种控件的消息处理.然后分别在各控件的消息处理函数里通过上面贴bitmap图片的方式来改变控件的外观.

         其实对于TCPMP程序UI部分的修改,最主要的还是要读懂源代码.在读懂源代码的基础上再结合Win32程序的结构来修改界面还是比较容易的.

目录
相关文章
|
2月前
|
算法 API C++
【Qt UI】QT 窗口/控件置顶方法详解
【Qt UI】QT 窗口/控件置顶方法详解
195 0
|
2月前
|
JSON 前端开发 测试技术
SAP UI5 sap.ui.core.util.MockServer.simulate 方法介绍
SAP UI5 sap.ui.core.util.MockServer.simulate 方法介绍
|
2月前
|
搜索推荐
SAP UI5 SmartTable.prototype._onMetadataInitialised 方法的深入介绍
SAP UI5 SmartTable.prototype._onMetadataInitialised 方法的深入介绍
|
2月前
|
JSON 前端开发 开发者
sap.ui.model.Model.checkUpdate 方法介绍
sap.ui.model.Model.checkUpdate 方法介绍
|
2月前
|
存储 JSON 供应链
sap.ui.model.Model.refresh 方法介绍
sap.ui.model.Model.refresh 方法介绍
|
1月前
|
存储 开发框架 JavaScript
深入探讨Flutter中动态UI构建的原理、方法以及数据驱动视图的实现技巧
【6月更文挑战第11天】Flutter是高效的跨平台移动开发框架,以其热重载、高性能渲染和丰富组件库著称。本文探讨了Flutter中动态UI构建原理与数据驱动视图的实现。动态UI基于Widget树模型,状态变化触发UI更新。状态管理是关键,Flutter提供StatefulWidget、Provider、Redux等方式。使用ListView等可滚动组件和StreamBuilder等流式组件实现数据驱动视图的自动更新。响应式布局确保UI在不同设备上的适应性。Flutter为开发者构建动态、用户友好的界面提供了强大支持。
36 2
|
14天前
|
JavaScript
vue + element UI【实战】音乐播放器/语音条(内含音频的加载、控制,事件监听,信息获取,手机网页阴影的去除等技巧)
vue + element UI【实战】音乐播放器/语音条(内含音频的加载、控制,事件监听,信息获取,手机网页阴影的去除等技巧)
13 1
|
2月前
|
小程序
【微信小程序6】引入第三方UI的方法(ColorUi)
【微信小程序6】引入第三方UI的方法(ColorUi)
172 0
|
2月前
|
Web App开发 存储 前端开发
谈谈企业级应用的自定义 UI 创建和集成方法一览
谈谈企业级应用的自定义 UI 创建和集成方法一览
|
8月前
|
开发者
SAP UI5 控件双向数据绑定后显示数据出问题,可以调试这个方法
SAP UI5 控件双向数据绑定后显示数据出问题,可以调试这个方法

相关实验场景

更多