里面涉及到很多知识

包括3D样式的去除,重绘ListView控件,以及处理控件的边框颜色

// Test_listview_1.cpp : Defines the entry point for the application.
// #include "stdafx.h"
#include "Test_listview_1.h"
#include <commctrl.h>
#include <uxtheme.h> #pragma comment (lib,"Comctl32.lib")
#pragma comment (lib,"UxTheme.lib")
//#pragma comment(linker,"\"/manifestdependency:type='win32' \
//name='Microsoft.Windows.Common-Controls' version='6.0.0.0' \
//processorArchitecture='*' publicKeyToken='6595b64144ccf1df' language='*'\"")
#define MAX_LOADSTRING 100 // Global Variables:
HINSTANCE hInst; // current instance
WCHAR szTitle[MAX_LOADSTRING]; // The title bar text
WCHAR szWindowClass[MAX_LOADSTRING]; // the main window class name
HWND hwndList1;
int middle = , middleH = , width = , height = ;
BOOL S_1 = FALSE;
// Forward declarations of functions included in this code module:
ATOM MyRegisterClass(HINSTANCE hInstance);
BOOL InitInstance(HINSTANCE, int);
LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);
LRESULT CALLBACK ListProc(HWND hwnd, UINT msg, WPARAM wp, LPARAM lp, UINT_PTR, DWORD_PTR);
INT_PTR CALLBACK About(HWND, UINT, WPARAM, LPARAM); int APIENTRY wWinMain(_In_ HINSTANCE hInstance,
_In_opt_ HINSTANCE hPrevInstance,
_In_ LPWSTR lpCmdLine,
_In_ int nCmdShow)
{
UNREFERENCED_PARAMETER(hPrevInstance);
UNREFERENCED_PARAMETER(lpCmdLine); // TODO: Place code here. // Initialize global strings
LoadStringW(hInstance, IDS_APP_TITLE, szTitle, MAX_LOADSTRING);
LoadStringW(hInstance, IDC_TESTLISTVIEW1, szWindowClass, MAX_LOADSTRING);
MyRegisterClass(hInstance); // Perform application initialization:
if (!InitInstance (hInstance, nCmdShow))
{
return FALSE;
} HACCEL hAccelTable = LoadAccelerators(hInstance, MAKEINTRESOURCE(IDC_TESTLISTVIEW1)); MSG msg; // Main message loop:
while (GetMessage(&msg, nullptr, , ))
{
if (!TranslateAccelerator(msg.hwnd, hAccelTable, &msg))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
} return (int) msg.wParam;
} //
// FUNCTION: MyRegisterClass()
//
// PURPOSE: Registers the window class.
//
ATOM MyRegisterClass(HINSTANCE hInstance)
{
WNDCLASSEXW wcex; wcex.cbSize = sizeof(WNDCLASSEX); wcex.style = CS_HREDRAW | CS_VREDRAW;
wcex.lpfnWndProc = WndProc;
wcex.cbClsExtra = ;
wcex.cbWndExtra = ;
wcex.hInstance = hInstance;
wcex.hIcon = LoadIcon(hInstance, MAKEINTRESOURCE(IDI_TESTLISTVIEW1));
wcex.hCursor = LoadCursor(nullptr, IDC_ARROW);
wcex.hbrBackground = (HBRUSH)(COLOR_WINDOW+);
wcex.lpszMenuName = MAKEINTRESOURCEW(IDC_TESTLISTVIEW1);
wcex.lpszClassName = szWindowClass;
wcex.hIconSm = LoadIcon(wcex.hInstance, MAKEINTRESOURCE(IDI_SMALL)); return RegisterClassExW(&wcex);
} //
// FUNCTION: InitInstance(HINSTANCE, int)
//
// PURPOSE: Saves instance handle and creates main window
//
// COMMENTS:
//
// In this function, we save the instance handle in a global variable and
// create and display the main program window.
//
BOOL InitInstance(HINSTANCE hInstance, int nCmdShow)
{
hInst = hInstance; // Store instance handle in our global variable HWND hWnd = CreateWindowW(szWindowClass, szTitle, WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT, , CW_USEDEFAULT, , nullptr, nullptr, hInstance, nullptr); if (!hWnd)
{
return FALSE;
} ShowWindow(hWnd, nCmdShow);
UpdateWindow(hWnd); return TRUE;
} //
// FUNCTION: WndProc(HWND, UINT, WPARAM, LPARAM)
//
// PURPOSE: Processes messages for the main window.
//
// WM_COMMAND - process the application menu
// WM_PAINT - Paint the main window
// WM_DESTROY - post a quit message and return
//
//
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
switch (message)
{
case WM_CREATE:
{
INITCOMMONCONTROLSEX icex; // Structure for control initialization.
icex.dwICC = ICC_LISTVIEW_CLASSES;
InitCommonControlsEx(&icex); hwndList1 = CreateWindow(WC_LISTVIEW, L"", WS_VISIBLE | WS_CHILD | LVS_REPORT | WS_BORDER | WS_VSCROLL | LVS_OWNERDRAWFIXED , , , width, height, hWnd, NULL, GetModuleHandle(NULL), );
DWORD Style = GetWindowLong(hwndList1, GWL_STYLE);
SetWindowLong(hwndList1, GWL_STYLE, Style &~WS_BORDER);
SetWindowSubclass(hwndList1, &ListProc, , NULL);
// HWND hHeader = ListView_GetHeader(hwndList1);
// SetWindowTheme(hHeader, L"", L"");
LVCOLUMN column;
column.mask = LVCF_WIDTH | LVCF_TEXT;
column.cx = ;
column.pszText = (LPWSTR)L"MASTER";
ListView_InsertColumn(hwndList1, , &column); //column for sub item 0 LVITEM lvi = {};
lvi.iItem = ListView_GetItemCount(hwndList1);
lvi.mask = LVIF_TEXT | LVIF_STATE;
lvi.pszText = (LPWSTR)L"MASTER1";
lvi.iSubItem = ;
ListView_InsertItem(hwndList1, &lvi);
lvi.pszText = (LPWSTR)L"MASTER2";
ListView_InsertItem(hwndList1, &lvi);
lvi.pszText = (LPWSTR)L"MASTER3";
ListView_InsertItem(hwndList1, &lvi);
lvi.pszText = (LPWSTR)L"MASTER4";
ListView_InsertItem(hwndList1, &lvi);
lvi.pszText = (LPWSTR)L"MASTER5";
ListView_InsertItem(hwndList1, &lvi);
lvi.pszText = (LPWSTR)L"MASTER6";
ListView_InsertItem(hwndList1, &lvi);
lvi.pszText = (LPWSTR)L"MASTER7";
ListView_InsertItem(hwndList1, &lvi);
}
break;
case WM_COMMAND:
{
int wmId = LOWORD(wParam);
// Parse the menu selections:
switch (wmId)
{
case IDM_ABOUT:
DialogBox(hInst, MAKEINTRESOURCE(IDD_ABOUTBOX), hWnd, About);
break;
case IDM_EXIT:
DestroyWindow(hWnd);
break;
default:
return DefWindowProc(hWnd, message, wParam, lParam);
}
}
break;
case WM_SIZE:
if (wParam == SIZE_RESTORED) {
SetWindowPos(hwndList1, , , , , , );
}
else if (wParam == SIZE_MAXIMIZED) {
SetWindowPos(hwndList1, , middle - ( + width), middleH - ( + height), width, height, );
RECT rc;
GetClientRect(hwndList1, &rc);
ListView_SetColumnWidth(hwndList1, , rc.right - rc.left);//rc.left is zero
}
break;
case WM_DRAWITEM:
{
LPDRAWITEMSTRUCT pDIS = (LPDRAWITEMSTRUCT)lParam;
HDC hDC = pDIS->hDC;
RECT rc = pDIS->rcItem;
HFONT hF;
HBRUSH bg = (HBRUSH)(::GetStockObject(DC_BRUSH));
HPEN pn = (HPEN)(::GetStockObject(NULL_PEN));
::SelectObject(hDC, bg);
::SelectObject(hDC, pn);
::SetTextColor(hDC, RGB(, , ));
int points = ;
// if (resX <= 800)
// {
hF = CreateFont(, , , , FW_NORMAL, FALSE, FALSE, FALSE, ANSI_CHARSET, OUT_DEFAULT_PRECIS, CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY, DEFAULT_PITCH | FF_SWISS, L"Tahoma");
// }
// else
// {
// points = 10;
// int fontheight = -MulDiv(points, GetDeviceCaps(hDC, LOGPIXELSY), 72);
// hF = CreateFont(fontheight, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, ANSI_CHARSET, OUT_DEFAULT_PRECIS, CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY, DEFAULT_PITCH | FF_SWISS, L"Tahoma");
// } HFONT hOldFont = (HFONT)SelectObject(hDC, hF);
if ((pDIS->itemID % ) != )
::SetDCBrushColor(hDC, RGB(, , ));
else {
::SetDCBrushColor(hDC, RGB(, , )); }
::Rectangle(hDC, rc.left, rc.top, rc.right, rc.bottom); wchar_t buffer[] = { };
ListView_GetItemText(pDIS->hwndItem, pDIS->itemID, , (LPWSTR)buffer, ); rc.left = ; ::DrawText(hDC, (LPWSTR)buffer, -, &rc, DT_SINGLELINE | DT_VCENTER);
SelectObject(hDC, hOldFont);
DeleteObject(hF); }
break;
case WM_MEASUREITEM:
{
MEASUREITEMSTRUCT * m = (MEASUREITEMSTRUCT*)lParam;
m->itemHeight = ;
}
break; //case WM_NOTIFY:
// if (((LPNMHDR)lParam)->code == NM_CUSTOMDRAW)
// {
// LPNMLVCUSTOMDRAW lplvcd = (LPNMLVCUSTOMDRAW)lParam;
// switch (lplvcd->nmcd.dwDrawStage)
// {
// case CDDS_PREPAINT:
// return CDRF_NOTIFYITEMDRAW; // case CDDS_ITEMPREPAINT:
// if (((int)lplvcd->nmcd.dwItemSpec % 2) == 0) {
// lplvcd->clrText = RGB(0, 0, 0);
// // lplvcd->clrTextBk = RGB(255, 255, 255);
// lplvcd->clrTextBk = RGB(193, 228, 255);
// }
// else {
// lplvcd->clrText = RGB(0, 0, 0);
// lplvcd->clrTextBk = RGB(202, 233, 255);
// } // return CDRF_NEWFONT;
// }
// return TRUE;
// }
//
case WM_PAINT:
{
PAINTSTRUCT ps; HDC hdc = BeginPaint(hWnd, &ps);
RECT rc,clrc;
GetWindowRect(hwndList1, &rc);
ScreenToClient(hWnd, (LPPOINT)&rc.left);
ScreenToClient(hWnd, (LPPOINT)&rc.right);
auto hpen = CreatePen(PS_SOLID, , RGB(, , ));
auto oldpen = SelectObject(hdc, hpen);
SelectObject(hdc, GetStockObject(NULL_BRUSH));
if (S_1)
{
Rectangle(hdc, rc.left - , rc.top - , rc.right + , rc.bottom + );
}
else
{
Rectangle(hdc, , , , );
}
SelectObject(hdc, oldpen);
DeleteObject(oldpen);
ReleaseDC(hwndList1, hdc);
// TODO: Add any drawing code that uses hdc here...
EndPaint(hWnd, &ps);
}
break; case WM_DESTROY:
PostQuitMessage();
break;
default:
return DefWindowProc(hWnd, message, wParam, lParam);
}
return ;
} // Message handler for about box.
INT_PTR CALLBACK About(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
{
UNREFERENCED_PARAMETER(lParam);
switch (message)
{
case WM_INITDIALOG:
return (INT_PTR)TRUE; case WM_COMMAND:
if (LOWORD(wParam) == IDOK || LOWORD(wParam) == IDCANCEL)
{
EndDialog(hDlg, LOWORD(wParam));
return (INT_PTR)TRUE;
}
break;
}
return (INT_PTR)FALSE;
} LRESULT CALLBACK ListProc(HWND hwnd, UINT msg, WPARAM wp, LPARAM lp, UINT_PTR, DWORD_PTR) {
switch (msg)
{
case WM_NOTIFY:
if (((LPNMHDR)lp)->code == NM_CUSTOMDRAW)
{
LPNMCUSTOMDRAW lpcd = (LPNMCUSTOMDRAW)lp;
switch (lpcd->dwDrawStage)
{
case CDDS_PREPAINT: return CDRF_NOTIFYITEMDRAW;
case CDDS_ITEMPREPAINT:
{ SetBkColor(lpcd->hdc, RGB(, , ));
// SetBkColor(lpcd->hdc, RGB(255, 0, 0));
SetTextColor(lpcd->hdc, RGB(, , ));
return CDRF_NEWFONT;
} }
} break; case WM_NCPAINT:
{
RECT rc;
GetWindowRect(hwnd, &rc);
OffsetRect(&rc, -rc.left, -rc.top);
auto hdc = GetWindowDC(hwnd);
auto hpen = CreatePen(PS_SOLID, , RGB(, , ));
auto oldpen = SelectObject(hdc, hpen);
SelectObject(hdc, GetStockObject(NULL_BRUSH));
// Rectangle(hdc, rc.left, rc.top, rc.right, rc.bottom);
SelectObject(hdc, oldpen);
DeleteObject(oldpen);
ReleaseDC(hwnd, hdc);
S_1 = TRUE;
// return 0;
break;
} case WM_NCDESTROY:
RemoveWindowSubclass(hwnd, ListProc, );
break;
} return DefSubclassProc(hwnd, msg, wp, lp);
} //InvalidateRect(hwnd, &rc, TRUE);

里面有很多注释的代码,我没删除是因为它们是有意义的

当创建ListView控件的时候,添加自定义的样式LVS_OWNERDRAWFIXED之后, 主窗口的WM_DRAWITEM起作用,WM_NOTIFY失效; 控件的自定义的窗口处理里的WM_NOTIFY依然有效。

ListView的创建的更多相关文章

  1. Android LIstView初次创建getview方法执行多次问题

    写listview优化的时候,发现Listview初次创建的时候会多次执行getView方法. <?xml version="1.0" encoding="utf- ...

  2. Android开发之自定义的ListView(UITableViewController)

    Android开发中的ListView, 顾名方法思义,就是表视图.表示图在iOS开发中就是TableView.两者虽然名称不一样,但是其使用方法,使用场景以及该控件的功能都极为相似,都是用来展示大量 ...

  3. Android 之 ListView的学习

    ListView 是一个控件,一个在垂直滚动的列表中显示条目的一个控件,这些条目的内容来自于一个ListAdapter .EditText Button TextView ImageView Chec ...

  4. Android之listview && adapter

    今天我们讲的也是非常重要的一个控件listview-最常用也是最难的 一个ListView通常有两个职责. (1)将数据填充到布局. (2)处理用户的选择点击等操作. 第一点很好理解,ListView ...

  5. 【转】ListView学习笔记(二)——ViewHolder

    在android开发中Listview是一个很重要的组件,它以列表的形式根据数据的长自适应展示具体内容,用户可以自由的定义listview每一列的布局,但当listview有大量的数据需要加载的时候, ...

  6. 【转】Android之自定义Adapter的ListView

    http://www.cnblogs.com/topcoderliu/archive/2011/05/07/2039862.html 在开发中,我们经常使用到ListView这个控件.Android的 ...

  7. Android listview与adapter用法

    listview与adapter用法 博客分类: android   一个ListView通常有两个职责. (1)将数据填充到布局. (2)处理用户的选择点击等操作. 第一点很好理解,ListView ...

  8. ListView 和 Adapter用法

    一个ListView通常有两个职责. (1)将数据填充到布局. (2)处理用户的选择点击等操作. 第一点很好理解,ListView就是实现这个功能的.第二点也不难做到,在后面的学习中读者会发现,这非常 ...

  9. Android中 ListView 详解(二)

    本文版权归 csdn noTice501 所有,转载请详细标明原作者及出处,以示尊重! 作者:noTice501 原文:http://blog.csdn.net/notice520/article/d ...

随机推荐

  1. 匿名内部类 this.val$的问题

    一天偶尔在网上找到一个jar包,反编译后出现了如下的代码: public void defineAnonymousInnerClass(String name)  {    new Thread(na ...

  2. wpf passwordbox控件 光标移到最后

    /// <summary> /// 设置光标位置 /// </summary> /// <param name="passwordBox">&l ...

  3. python文档的数据读取,把读取数据写入到新的表里

    目的:接口自动化过程需要从表格文件读取,然后把结果写到表格中.没有多余内容全部是精华! 读取文件1 读取文件2 代码如下图: # -*-coding:utf-8 -*-# Author:wangjun ...

  4. 【Qt开发】在QLabel已经显示背景图片后绘制图形注意事项

    主要是要解决图形覆盖的问题,通常的办法就是对QLabel进行子类化,并重载函数: void myLabel::paintEvent(QPaintEvent *event)   {       QLab ...

  5. javascript 数据类型之数值转换

    数值转换 一.有3个函数可以把非数值转换为数值: Number() parse Int() parse Float() 说明: 1.Number()可以用于任何数据类型,强转类型,如果不能把指转成数值 ...

  6. 【7.10校内test】T2不等数列

    [题目链接luogu] 此题在luogu上模数是2015,考试题的模数是2012. 然后这道题听说好多人是打表找规律的(就像7.9T2一样)(手动滑稽_gc) 另外手动 sy,每次测试都无意之间bib ...

  7. mysql之general log 日志

    开启 general log 将所有到达MySQL Server的SQL语句记录下来. 一般不会开启开功能,因为log的量会非常庞大.但个别情况下可能会临时的开一会儿general log以供排障使用 ...

  8. Inversions After Shuffle CodeForces - 749E (概率,期望)

    大意: 给定一个$n$排列, 随机选一个区间, 求将区间随机重排后整个序列的逆序对期望. 考虑对区间$[l,r]$重排后逆序对的变化, 显然只有区间[l,r]内部会发生改变 而长为$k$的随机排列期望 ...

  9. 运维DNS原理配置

    Linux DNS原理简介及配置 DNS简介 DNS原理 域名解析的过程 资源记录 DNS BIND安装配置 一.简介 一般来讲域名比IP地址更加的有含义.也更容易记住,所以通常用户更习惯输入域名来访 ...

  10. ES中的查询操作

    1.前缀查询 先输入数据: PUT /my_index/address/ { "postcode": "W1 3DG" } PUT /my_index/addr ...