#include <windows.h>
#include <commctrl.h>

#include "drag.h"
#include "listview.h"
#include "win32.h"
#include "window.h"

/* Actual window procedure for all list views, which calls the
 * appropriate WndProc member function. */
static LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);

ListView::ListView(Window& parent, const HMENU hMenu, const DWORD dwStyle) : parent(parent)
{
	extern HFONT g_hfNormal;

	hWnd = Require(CreateWindowExW(
	    WS_EX_CLIENTEDGE,
	    WC_LISTVIEW,
	    L"",
	    dwStyle|WS_CHILD|WS_VISIBLE|WS_VSCROLL|WS_TABSTOP|LVS_REPORT|LVS_SHOWSELALWAYS,
	    0, 0, 0, 0,
	    parent.hWnd, hMenu, GetModuleHandle(nullptr), this));
	m_proc0 = reinterpret_cast<WNDPROC>(SetWindowLongPtr(
	    hWnd, GWLP_WNDPROC, reinterpret_cast<LONG_PTR>(::WndProc)));

	Require(SetPropW(hWnd, L"this", reinterpret_cast<HANDLE>(this)));
	ListView_SetExtendedListViewStyle(hWnd, LVS_EX_FULLROWSELECT|LVS_EX_DOUBLEBUFFER);
	SendMessageW(hWnd, WM_SETFONT, reinterpret_cast<WPARAM>(g_hfNormal), MAKELPARAM(FALSE, 0));
}

int ListView::Height()
{
	const int cItem = ListView_GetItemCount(hWnd);
	return Dpi(4)+cItem*Dpi(19);
}

void ListView::ResizeColumns(int) {}

void ListView::UpdateTheme(const bool bThemeActive)
{
	const wchar_t* theme;
	WORD action;
	extern HRESULT (__stdcall *SetWindowTheme)(HWND, const wchar_t*, const wchar_t*);

	if (!SetWindowTheme) return;
	if (bThemeActive) {
		theme = L"Explorer";
		action = UIS_SET;
	} else {
		theme = nullptr;
		action = UIS_CLEAR;
	}

	/* Use modern "Explorer" theme. */
	SetWindowTheme(hWnd, theme, nullptr);

	/* Hide focus rectangles. */
	SendMessageW(hWnd, WM_UPDATEUISTATE, MAKEWPARAM(action, UISF_HIDEFOCUS), 0);
}

LRESULT CALLBACK ListView::WndProc(const HWND hWnd, const UINT uMsg,
    const WPARAM wParam, const LPARAM lParam)
{
	switch (uMsg) {
	case WM_NOTIFY:
		switch (reinterpret_cast<NMHDR*>(lParam)->code) {
		case HDN_ENDTRACK:
			parent.UpdateLayout();
			return TRUE;
		}
		break;
	case WM_LBUTTONDOWN:
	case WM_NCLBUTTONDOWN:
		if (parent.dragDlv.HandleLButtonDown())
			return 0;
		break;
	}

	return CallWindowProc(m_proc0, hWnd, uMsg, wParam, lParam);
}

LRESULT CALLBACK WndProc(const HWND hWnd, const UINT uMsg, const WPARAM wParam,
    const LPARAM lParam)
{
	ListView* const lv = reinterpret_cast<ListView*>(GetProp(hWnd, L"this"));

	if (uMsg == WM_DESTROY)
		RemovePropW(hWnd, L"this");

	return lv? lv->WndProc(hWnd, uMsg, wParam, lParam): FALSE;
}