reactos/dll/win32/browseui/CAutoComplete.cpp
Katayama Hirofumi MZ 773ad7aebc
[BROWSEUI] auto-completion: Support large number items (#3592)
If the items are too many, enable filtering in item enumeration. CORE-9281
2021-04-08 15:02:05 +09:00

2118 lines
61 KiB
C++

/*
* AutoComplete interfaces implementation.
*
* Copyright 2004 Maxime Bellengé <maxime.bellenge@laposte.net>
* Copyright 2009 Andrew Hill
* Copyright 2020-2021 Katayama Hirofumi MZ <katayama.hirofumi.mz@gmail.com>
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
*/
#include "precomp.h"
/*
TODO:
- implement ACO_SEARCH style
- implement ACO_FILTERPREFIXES style
- implement ACO_RTLREADING style
*/
#define CX_LIST 30160 // width of m_hwndList (very wide but alright)
#define CY_LIST 288 // maximum height of drop-down window
#define CY_ITEM 18 // default height of listview item
#define COMPLETION_TIMEOUT 300 // in milliseconds
#define MAX_ITEM_COUNT 1000 // the maximum number of items
#define WATCH_TIMER_ID 0xFEEDBEEF // timer ID to watch m_rcEdit
#define WATCH_INTERVAL 300 // in milliseconds
static HHOOK s_hMouseHook = NULL; // hook handle
static HWND s_hWatchWnd = NULL; // the window handle to watch
struct PREFIX_INFO
{
LPCWSTR psz;
INT cch;
};
static const PREFIX_INFO s_prefixes[] =
{
{ L"https://", 8 },
{ L"http://www.", 11 },
{ L"http://", 7 },
{ L"www.", 4 },
};
static inline BOOL DropPrefix(const CStringW& str, CStringW& strBody)
{
for (size_t iPrefix = 0; iPrefix < _countof(s_prefixes); ++iPrefix)
{
LPCWSTR psz = s_prefixes[iPrefix].psz;
INT cch = s_prefixes[iPrefix].cch;
if (::StrCmpNIW(str, psz, cch) == 0)
{
strBody = str.Mid(cch);
return TRUE;
}
}
strBody = str;
return FALSE;
}
// mouse hook procedure to watch the mouse click
// https://docs.microsoft.com/en-us/previous-versions/windows/desktop/legacy/ms644988(v=vs.85)
static LRESULT CALLBACK MouseProc(INT nCode, WPARAM wParam, LPARAM lParam)
{
if (s_hMouseHook == NULL)
return 0; // do default
// if the user clicked the outside of s_hWatchWnd, then hide the drop-down window
if (nCode == HC_ACTION && // an action?
s_hWatchWnd && ::IsWindow(s_hWatchWnd) && // s_hWatchWnd is valid?
::GetCapture() == NULL) // no capture? (dragging something?)
{
RECT rc;
MOUSEHOOKSTRUCT *pMouseHook = reinterpret_cast<MOUSEHOOKSTRUCT *>(lParam);
switch (wParam)
{
case WM_LBUTTONDOWN: case WM_LBUTTONUP:
case WM_RBUTTONDOWN: case WM_RBUTTONUP:
case WM_MBUTTONDOWN: case WM_MBUTTONUP:
case WM_NCLBUTTONDOWN: case WM_NCLBUTTONUP:
case WM_NCRBUTTONDOWN: case WM_NCRBUTTONUP:
case WM_NCMBUTTONDOWN: case WM_NCMBUTTONUP:
{
::GetWindowRect(s_hWatchWnd, &rc);
if (!::PtInRect(&rc, pMouseHook->pt)) // outside of s_hWatchWnd?
{
::ShowWindowAsync(s_hWatchWnd, SW_HIDE); // hide it
}
break;
}
}
}
return ::CallNextHookEx(s_hMouseHook, nCode, wParam, lParam); // go next hook
}
//////////////////////////////////////////////////////////////////////////////
// sorting algorithm
// http://www.ics.kagoshima-u.ac.jp/~fuchida/edu/algorithm/sort-algorithm/
typedef CSimpleArray<CStringW> list_t;
static inline INT compare1(const CStringW& str1, const CStringW& str2)
{
CStringW s1, s2;
DropPrefix(str1, s1);
DropPrefix(str2, s2);
return s1.CompareNoCase(s2);
}
static inline INT pivot(list_t& list, INT i, INT j)
{
INT k = i + 1;
while (k <= j && compare1(list[i], list[k]) == 0)
k++;
if (k > j)
return -1;
if (compare1(list[i], list[k]) >= 0)
return i;
return k;
}
static inline INT partition(list_t& list, INT i, INT j, const CStringW& x)
{
INT left = i, right = j;
while (left <= right)
{
while (left <= j && compare1(list[left], x) < 0)
left++;
while (right >= i && compare1(list[right], x) >= 0)
right--;
if (left > right)
break;
CStringW tmp = list[left];
list[left] = list[right];
list[right] = tmp;
left++;
right--;
}
return left;
}
static void quicksort(list_t& list, INT i, INT j)
{
if (i == j)
return;
INT p = pivot(list, i, j);
if (p == -1)
return;
INT k = partition(list, i, j, list[p]);
quicksort(list, i, k - 1);
quicksort(list, k, j);
}
static inline void DoSort(list_t& list)
{
if (list.GetSize() <= 1) // sanity check
return;
quicksort(list, 0, list.GetSize() - 1); // quick sort
}
// std::unique
static INT DoUnique(list_t& list)
{
INT first = 0, last = list.GetSize();
if (first == last)
return last;
INT result = first;
while (++first != last)
{
if (compare1(list[result], list[first]) != 0)
list[++result] = list[first];
}
return ++result;
}
static inline void DoUniqueAndTrim(list_t& list)
{
INT last = DoUnique(list);
while (list.GetSize() > last)
{
list.RemoveAt(last);
}
}
//////////////////////////////////////////////////////////////////////////////
// range of WCHAR (inclusive)
struct RANGE
{
WCHAR from, to;
};
// a callback function for bsearch: comparison of two ranges
static inline int RangeCompare(const void *x, const void *y)
{
const RANGE *a = reinterpret_cast<const RANGE *>(x);
const RANGE *b = reinterpret_cast<const RANGE *>(y);
if (a->to < b->from)
return -1;
if (b->to < a->from)
return 1;
return 0;
}
// is the WCHAR a word break?
static inline BOOL IsWordBreak(WCHAR ch)
{
// the ranges of word break characters
static const RANGE s_ranges[] =
{
{ 0x0009, 0x0009 }, { 0x0020, 0x002f }, { 0x003a, 0x0040 }, { 0x005b, 0x0060 },
{ 0x007b, 0x007e }, { 0x00ab, 0x00ab }, { 0x00ad, 0x00ad }, { 0x00bb, 0x00bb },
{ 0x02c7, 0x02c7 }, { 0x02c9, 0x02c9 }, { 0x055d, 0x055d }, { 0x060c, 0x060c },
{ 0x2002, 0x200b }, { 0x2013, 0x2014 }, { 0x2016, 0x2016 }, { 0x2018, 0x2018 },
{ 0x201c, 0x201d }, { 0x2022, 0x2022 }, { 0x2025, 0x2027 }, { 0x2039, 0x203a },
{ 0x2045, 0x2046 }, { 0x207d, 0x207e }, { 0x208d, 0x208e }, { 0x226a, 0x226b },
{ 0x2574, 0x2574 }, { 0x3001, 0x3003 }, { 0x3005, 0x3005 }, { 0x3008, 0x3011 },
{ 0x3014, 0x301b }, { 0x301d, 0x301e }, { 0x3041, 0x3041 }, { 0x3043, 0x3043 },
{ 0x3045, 0x3045 }, { 0x3047, 0x3047 }, { 0x3049, 0x3049 }, { 0x3063, 0x3063 },
{ 0x3083, 0x3083 }, { 0x3085, 0x3085 }, { 0x3087, 0x3087 }, { 0x308e, 0x308e },
{ 0x309b, 0x309e }, { 0x30a1, 0x30a1 }, { 0x30a3, 0x30a3 }, { 0x30a5, 0x30a5 },
{ 0x30a7, 0x30a7 }, { 0x30a9, 0x30a9 }, { 0x30c3, 0x30c3 }, { 0x30e3, 0x30e3 },
{ 0x30e5, 0x30e5 }, { 0x30e7, 0x30e7 }, { 0x30ee, 0x30ee }, { 0x30f5, 0x30f6 },
{ 0x30fc, 0x30fe }, { 0xfd3e, 0xfd3f }, { 0xfe30, 0xfe31 }, { 0xfe33, 0xfe44 },
{ 0xfe4f, 0xfe51 }, { 0xfe59, 0xfe5e }, { 0xff08, 0xff09 }, { 0xff0c, 0xff0c },
{ 0xff0e, 0xff0e }, { 0xff1c, 0xff1c }, { 0xff1e, 0xff1e }, { 0xff3b, 0xff3b },
{ 0xff3d, 0xff3d }, { 0xff40, 0xff40 }, { 0xff5b, 0xff5e }, { 0xff61, 0xff64 },
{ 0xff67, 0xff70 }, { 0xff9e, 0xff9f }, { 0xffe9, 0xffe9 }, { 0xffeb, 0xffeb },
};
// binary search
RANGE range = { ch, ch };
return !!bsearch(&range, s_ranges, _countof(s_ranges), sizeof(RANGE), RangeCompare);
}
// This function is an application-defined callback function.
// https://docs.microsoft.com/en-us/windows/win32/api/winuser/nc-winuser-editwordbreakprocw
static INT CALLBACK
EditWordBreakProcW(LPWSTR lpch, INT index, INT count, INT code)
{
switch (code)
{
case WB_ISDELIMITER:
return IsWordBreak(lpch[index]);
case WB_LEFT:
{
if (index)
--index;
while (index && !IsWordBreak(lpch[index]))
--index;
return index;
}
case WB_RIGHT:
{
if (!count)
break;
while (index < count && lpch[index] && !IsWordBreak(lpch[index]))
++index;
return index;
}
}
return 0;
}
static LRESULT CALLBACK
EditSubclassProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam,
UINT_PTR uSubclassID, DWORD_PTR dwData)
{
CAutoComplete *pThis = reinterpret_cast<CAutoComplete *>(dwData);
return pThis->EditWndProc(hwnd, uMsg, wParam, lParam);
}
LRESULT CAutoComplete::EditWndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
LRESULT ret;
HWND hwndGotFocus;
switch (uMsg)
{
case WM_CHAR:
return OnEditChar(wParam, lParam);
case WM_CUT: case WM_PASTE: case WM_CLEAR:
ret = ::DefSubclassProc(hwnd, uMsg, wParam, lParam); // do default
UpdateCompletion(TRUE);
return ret;
case WM_GETDLGCODE:
ret = ::DefSubclassProc(hwnd, uMsg, wParam, lParam); // do default
// some special keys need default processing. we handle them here
switch (wParam)
{
case VK_RETURN:
if (IsWindowVisible() || ::GetKeyState(VK_CONTROL) < 0)
OnEditKeyDown(VK_RETURN, 0);
break;
case VK_TAB:
if (IsWindowVisible() && UseTab())
ret |= DLGC_WANTALLKEYS; // we want all keys to manipulate the list
break;
case VK_ESCAPE:
if (IsWindowVisible())
ret |= DLGC_WANTALLKEYS; // we want all keys to manipulate the list
break;
default:
{
ret |= DLGC_WANTALLKEYS; // we want all keys to manipulate the list
break;
}
}
return ret;
case WM_KEYDOWN:
if (OnEditKeyDown(wParam, lParam))
return TRUE; // eat
break;
case WM_SETFOCUS:
break;
case WM_KILLFOCUS:
// hide the list if lost focus
hwndGotFocus = (HWND)wParam;
if (hwndGotFocus != m_hwndEdit && hwndGotFocus != m_hWnd)
HideDropDown();
break;
case WM_SETTEXT:
if (!m_bInSetText)
HideDropDown(); // it's mechanical WM_SETTEXT
break;
case WM_DESTROY:
{
if (m_fnOldWordBreakProc)
{
::SendMessageW(hwnd, EM_SETWORDBREAKPROC, 0, reinterpret_cast<LPARAM>(m_fnOldWordBreakProc));
m_fnOldWordBreakProc = NULL;
}
::RemoveWindowSubclass(hwnd, EditSubclassProc, 0);
if (::IsWindow(m_hWnd))
PostMessageW(WM_CLOSE, 0, 0);
// remove reference to m_hwndEdit
Release();
break;
}
}
return ::DefSubclassProc(hwnd, uMsg, wParam, lParam); // do default
}
//////////////////////////////////////////////////////////////////////////////
// CACListView
CACListView::CACListView() : m_pDropDown(NULL), m_cyItem(CY_ITEM)
{
}
HWND CACListView::Create(HWND hwndParent)
{
ATLASSERT(m_hWnd == NULL);
DWORD dwStyle = WS_CHILD | /*WS_VISIBLE |*/ WS_CLIPSIBLINGS | LVS_NOCOLUMNHEADER |
LVS_OWNERDATA | LVS_OWNERDRAWFIXED | LVS_SINGLESEL | LVS_REPORT;
HWND hWnd = ::CreateWindowExW(0, GetWndClassName(), L"Internet Explorer", dwStyle,
0, 0, 0, 0, hwndParent, NULL,
_AtlBaseModule.GetModuleInstance(), NULL);
SubclassWindow(hWnd); // do subclass to handle messages
// set extended listview style
DWORD exstyle = LVS_EX_ONECLICKACTIVATE | LVS_EX_FULLROWSELECT | LVS_EX_TRACKSELECT;
SetExtendedListViewStyle(exstyle, exstyle);
// insert one column (needed to insert items)
LV_COLUMNW column = { LVCF_FMT | LVCF_WIDTH };
column.fmt = LVCFMT_LEFT;
column.cx = CX_LIST - ::GetSystemMetrics(SM_CXVSCROLL);
InsertColumn(0, &column);
return m_hWnd;
}
// set font handle
VOID CACListView::SetFont(HFONT hFont)
{
SendMessageW(WM_SETFONT, (WPARAM)hFont, TRUE); // set font
// get listview item height
m_cyItem = CY_ITEM;
HDC hDC = GetDC();
if (hDC)
{
HGDIOBJ hFontOld = ::SelectObject(hDC, hFont);
TEXTMETRICW tm;
if (::GetTextMetricsW(hDC, &tm))
{
m_cyItem = (tm.tmHeight * 3) / 2; // 3/2 of text height
}
::SelectObject(hDC, hFontOld);
ReleaseDC(hDC);
}
}
// get the number of visible items
INT CACListView::GetVisibleCount()
{
if (m_cyItem <= 0) // avoid "division by zero"
return 0;
CRect rc;
GetClientRect(&rc);
return rc.Height() / m_cyItem;
}
// get the text of an item
CStringW CACListView::GetItemText(INT iItem)
{
// NOTE: LVS_OWNERDATA doesn't support LVM_GETITEMTEXT.
ATLASSERT(m_pDropDown);
ATLASSERT(GetStyle() & LVS_OWNERDATA);
return m_pDropDown->GetItemText(iItem);
}
// get the item index from position
INT CACListView::ItemFromPoint(INT x, INT y)
{
LV_HITTESTINFO hittest;
hittest.pt.x = x;
hittest.pt.y = y;
return HitTest(&hittest);
}
// get current selection
INT CACListView::GetCurSel()
{
return GetNextItem(-1, LVNI_ALL | LVNI_SELECTED);
}
// set current selection
VOID CACListView::SetCurSel(INT iItem)
{
if (iItem == -1)
SetItemState(-1, 0, LVIS_SELECTED); // select none
else
SetItemState(iItem, LVIS_SELECTED, LVIS_SELECTED);
}
// select the specific position (in client coordinates)
VOID CACListView::SelectHere(INT x, INT y)
{
SetCurSel(ItemFromPoint(x, y));
}
// WM_LBUTTONUP / WM_MBUTTONUP / WM_RBUTTONUP @implemented
LRESULT CACListView::OnButtonUp(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
TRACE("CACListView::OnButtonUp(%p)\n", this);
return 0; // eat
}
// WM_LBUTTONDOWN @implemented
// This message is posted when the user pressed the left mouse button while the cursor is inside.
LRESULT CACListView::OnLButtonDown(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
TRACE("CACListView::OnLButtonDown(%p)\n", this);
ATLASSERT(m_pDropDown);
INT iItem = ItemFromPoint(GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam));
if (iItem != -1)
{
m_pDropDown->SelectItem(iItem); // select the item
CStringW strText = GetItemText(iItem); // get text of item
m_pDropDown->SetEditText(strText); // set text
m_pDropDown->SetEditSel(0, strText.GetLength()); // select all
m_pDropDown->HideDropDown(); // hide
}
return 0;
}
// WM_MBUTTONDOWN / WM_RBUTTONDOWN @implemented
LRESULT CACListView::OnMRButtonDown(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
TRACE("CACListView::OnMRButtonDown(%p)\n", this);
return 0; // eat
}
// WM_MOUSEWHEEL @implemented
LRESULT CACListView::OnMouseWheel(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
TRACE("CACListView::OnMouseWheel(%p)\n", this);
ATLASSERT(m_pDropDown);
LRESULT ret = DefWindowProcW(uMsg, wParam, lParam); // do default
m_pDropDown->UpdateScrollBar();
return ret;
}
// WM_NCHITTEST
LRESULT CACListView::OnNCHitTest(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
TRACE("CACListView::OnNCHitTest(%p)\n", this);
ATLASSERT(m_pDropDown);
POINT pt = { GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam) }; // in screen coordinates
ScreenToClient(&pt); // into client coordinates
HWND hwndTarget = m_pDropDown->ChildWindowFromPoint(pt);
if (hwndTarget != m_hWnd)
return HTTRANSPARENT; // pass through (for resizing the drop-down window)
bHandled = FALSE; // do default
return 0;
}
//////////////////////////////////////////////////////////////////////////////
// CACScrollBar
HWND CACScrollBar::Create(HWND hwndParent)
{
ATLASSERT(m_hWnd == NULL);
DWORD dwStyle = WS_CHILD | /*WS_VISIBLE |*/ SBS_BOTTOMALIGN | SBS_VERT;
m_hWnd = ::CreateWindowExW(0, GetWndClassName(), NULL, dwStyle,
0, 0, 0, 0, hwndParent, NULL,
_AtlBaseModule.GetModuleInstance(), NULL);
// we don't subclass because no message handling is needed
return m_hWnd;
}
//////////////////////////////////////////////////////////////////////////////
// CACSizeBox
HWND CACSizeBox::Create(HWND hwndParent)
{
ATLASSERT(m_hWnd == NULL);
DWORD dwStyle = WS_CHILD | /*WS_VISIBLE |*/ SBS_SIZEBOX;
HWND hWnd = ::CreateWindowExW(0, GetWndClassName(), NULL, dwStyle,
0, 0, 0, 0, hwndParent, NULL,
_AtlBaseModule.GetModuleInstance(), NULL);
SubclassWindow(hWnd); // do subclass to handle message
return m_hWnd;
}
VOID CACSizeBox::SetStatus(BOOL bDowner, BOOL bLongList)
{
// set flags
m_bDowner = bDowner;
m_bLongList = bLongList;
if (bLongList)
{
SetWindowRgn(NULL, TRUE); // reset window region
return;
}
RECT rc;
GetWindowRect(&rc); // get size-box size
::OffsetRect(&rc, -rc.left, -rc.top); // window regions use special coordinates
ATLASSERT(rc.left == 0 && rc.top == 0);
// create a trianglar region
HDC hDC = ::CreateCompatibleDC(NULL);
::BeginPath(hDC);
if (m_bDowner)
{
::MoveToEx(hDC, rc.right, 0, NULL);
::LineTo(hDC, rc.right, rc.bottom);
::LineTo(hDC, 0, rc.bottom);
::LineTo(hDC, rc.right, 0);
}
else
{
::MoveToEx(hDC, rc.right, rc.bottom, NULL);
::LineTo(hDC, rc.right, 0);
::LineTo(hDC, 0, 0);
::LineTo(hDC, rc.right, rc.bottom);
}
::EndPath(hDC);
HRGN hRgn = ::PathToRegion(hDC);
::DeleteDC(hDC);
SetWindowRgn(hRgn, TRUE); // set the trianglar region
}
// WM_ERASEBKGND
LRESULT CACSizeBox::OnEraseBkGnd(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
return TRUE; // do nothing (for quick drawing)
}
// WM_NCHITTEST
LRESULT CACSizeBox::OnNCHitTest(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
return HTTRANSPARENT; // pass through
}
// WM_PAINT
LRESULT CACSizeBox::OnPaint(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
CRect rc;
GetClientRect(&rc);
PAINTSTRUCT ps;
HDC hDC = BeginPaint(&ps);
if (!hDC)
return 0;
// fill background
::FillRect(hDC, &rc, ::GetSysColorBrush(COLOR_3DFACE));
// draw size-box
INT cxy = rc.Width();
for (INT shift = 0; shift < 2; ++shift)
{
// choose pen color
INT iColor = ((shift == 0) ? COLOR_HIGHLIGHTTEXT : COLOR_3DSHADOW);
HPEN hPen = ::CreatePen(PS_SOLID, 1, ::GetSysColor(iColor));
HGDIOBJ hPenOld = ::SelectObject(hDC, hPen);
// do loop to draw the slanted lines
for (INT delta = cxy / 4; delta < cxy; delta += cxy / 4)
{
// draw a grip line
if (m_bDowner)
{
::MoveToEx(hDC, rc.right, rc.top + delta + shift, NULL);
::LineTo(hDC, rc.left + delta + shift, rc.bottom);
}
else
{
::MoveToEx(hDC, rc.left + delta + shift, rc.top, NULL);
::LineTo(hDC, rc.right, rc.bottom - delta - shift);
}
}
// delete pen
::SelectObject(hDC, hPenOld);
::DeleteObject(hPen);
}
EndPaint(&ps);
return 0;
}
//////////////////////////////////////////////////////////////////////////////
// CAutoComplete public methods
CAutoComplete::CAutoComplete()
: m_bInSetText(FALSE), m_bInSelectItem(FALSE)
, m_bDowner(TRUE), m_dwOptions(ACO_AUTOAPPEND | ACO_AUTOSUGGEST)
, m_bEnabled(TRUE), m_hwndCombo(NULL), m_hFont(NULL), m_bResized(FALSE)
, m_hwndEdit(NULL), m_fnOldEditProc(NULL), m_fnOldWordBreakProc(NULL)
, m_bPartialList(FALSE), m_dwTick(0)
{
}
HWND CAutoComplete::CreateDropDown()
{
ATLASSERT(m_hWnd == NULL);
DWORD dwStyle = WS_POPUP | /*WS_VISIBLE |*/ WS_CLIPSIBLINGS | WS_CLIPCHILDREN | WS_BORDER;
DWORD dwExStyle = WS_EX_TOOLWINDOW | WS_EX_TOPMOST | WS_EX_NOPARENTNOTIFY;
Create(NULL, NULL, NULL, dwStyle, dwExStyle);
TRACE("CAutoComplete::CreateDropDown(%p): m_hWnd=%p, m_hwndEdit=%p\n",
this, m_hWnd, m_hwndEdit);
return m_hWnd;
}
CAutoComplete::~CAutoComplete()
{
TRACE("CAutoComplete::~CAutoComplete(%p)\n", this);
if (m_hFont)
{
::DeleteObject(m_hFont);
m_hFont = NULL;
}
// quit holding them
m_pEnum.Release();
m_pACList.Release();
}
inline BOOL CAutoComplete::CanAutoSuggest() const
{
return !!(m_dwOptions & ACO_AUTOSUGGEST) && m_bEnabled;
}
inline BOOL CAutoComplete::CanAutoAppend() const
{
return !!(m_dwOptions & ACO_AUTOAPPEND) && m_bEnabled;
}
inline BOOL CAutoComplete::UseTab() const
{
return !!(m_dwOptions & ACO_USETAB) && m_bEnabled;
}
inline BOOL CAutoComplete::IsComboBoxDropped() const
{
if (!::IsWindow(m_hwndCombo))
return FALSE;
return (BOOL)::SendMessageW(m_hwndCombo, CB_GETDROPPEDSTATE, 0, 0);
}
inline BOOL CAutoComplete::FilterPrefixes() const
{
return !!(m_dwOptions & ACO_FILTERPREFIXES) && m_bEnabled;
}
inline INT CAutoComplete::GetItemCount() const
{
return m_outerList.GetSize();
}
CStringW CAutoComplete::GetItemText(INT iItem) const
{
if (iItem < 0 || m_outerList.GetSize() <= iItem)
return L"";
return m_outerList[iItem];
}
inline CStringW CAutoComplete::GetEditText() const
{
WCHAR szText[L_MAX_URL_LENGTH];
if (::GetWindowTextW(m_hwndEdit, szText, _countof(szText)))
return szText;
return L"";
}
VOID CAutoComplete::SetEditText(LPCWSTR pszText)
{
m_bInSetText = TRUE; // don't hide drop-down
::CallWindowProcW(m_fnOldEditProc, m_hwndEdit, WM_SETTEXT, 0, reinterpret_cast<LPARAM>(pszText));
m_bInSetText = FALSE;
}
inline CStringW CAutoComplete::GetStemText(const CStringW& strText) const
{
INT ich = strText.ReverseFind(L'\\');
if (ich == -1)
return L""; // no stem
return strText.Left(ich + 1);
}
VOID CAutoComplete::SetEditSel(INT ich0, INT ich1)
{
::CallWindowProcW(m_fnOldEditProc, m_hwndEdit, EM_SETSEL, ich0, ich1);
}
VOID CAutoComplete::ShowDropDown()
{
if (!m_hWnd || !CanAutoSuggest())
return;
INT cItems = GetItemCount();
if (cItems == 0 || ::GetFocus() != m_hwndEdit || IsComboBoxDropped())
{
// hide the drop-down if necessary
HideDropDown();
return;
}
RepositionDropDown();
}
VOID CAutoComplete::HideDropDown()
{
ShowWindow(SW_HIDE);
}
VOID CAutoComplete::SelectItem(INT iItem)
{
m_hwndList.SetCurSel(iItem);
if (iItem != -1)
m_hwndList.EnsureVisible(iItem, FALSE);
}
VOID CAutoComplete::DoAutoAppend()
{
if (!CanAutoAppend()) // can we auto-append?
return; // don't append
CStringW strText = GetEditText(); // get the text
if (strText.IsEmpty())
return; // don't append
INT cItems = m_innerList.GetSize(); // get the number of items
if (cItems == 0)
return; // don't append
// get the common string
CStringW strCommon;
BOOL bFound = FALSE;
for (INT iItem = 0; iItem < cItems; ++iItem)
{
const CStringW& strItem = m_innerList[iItem]; // get the text of the item
CStringW strBody;
if (DropPrefix(strItem, strBody) &&
::StrCmpNIW(strBody, strText, strText.GetLength()) == 0)
{
if (!bFound)
{
bFound = TRUE;
strCommon = strBody;
continue;
}
for (INT ich = 0; ich < strBody.GetLength(); ++ich)
{
if (strCommon.GetLength() <= ich)
break;
if (ChrCmpIW(strCommon[ich], strBody[ich]) != 0)
{
strCommon = strCommon.Left(ich);
break;
}
}
continue;
}
if (::StrCmpNIW(strItem, strText, strText.GetLength()) == 0)
{
if (!bFound)
{
bFound = TRUE;
strCommon = strItem;
continue;
}
for (INT ich = 0; ich < strItem.GetLength(); ++ich)
{
if (strCommon.GetLength() <= ich)
break;
if (ChrCmpIW(strCommon[ich], strItem[ich]) != 0)
{
strCommon = strCommon.Left(ich);
break;
}
}
}
}
if (strCommon.IsEmpty() || strCommon.GetLength() <= strText.GetLength())
return; // no suggestion
// append suggestion
SetEditText(strCommon);
// select the appended suggestion
SetEditSel(strText.GetLength(), strCommon.GetLength());
}
// go back a word ([Ctrl]+[Backspace])
VOID CAutoComplete::DoBackWord()
{
// get current selection
INT ich0, ich1;
::CallWindowProcW(m_fnOldEditProc, m_hwndEdit, EM_GETSEL,
reinterpret_cast<WPARAM>(&ich0), reinterpret_cast<LPARAM>(&ich1));
if (ich0 <= 0 || ich0 != ich1) // there is selection or left-side end
return; // don't do anything
// get text
CStringW str = GetEditText();
// extend the range
while (ich0 > 0 && IsWordBreak(str[ich0 - 1]))
--ich0;
while (ich0 > 0 && !IsWordBreak(str[ich0 - 1]))
--ich0;
// select range
SetEditSel(ich0, ich1);
// replace selection with empty text (this is actually deletion)
::CallWindowProcW(m_fnOldEditProc, m_hwndEdit, EM_REPLACESEL, TRUE, (LPARAM)L"");
}
VOID CAutoComplete::UpdateScrollBar()
{
// copy scroll info from m_hwndList to m_hwndScrollBar
SCROLLINFO si = { sizeof(si), SIF_ALL };
m_hwndList.GetScrollInfo(SB_VERT, &si);
m_hwndScrollBar.SetScrollInfo(SB_CTL, &si, FALSE);
// show/hide scroll bar
INT cVisibles = m_hwndList.GetVisibleCount();
INT cItems = m_hwndList.GetItemCount();
BOOL bShowScroll = (cItems > cVisibles);
m_hwndScrollBar.ShowWindow(bShowScroll ? SW_SHOWNOACTIVATE : SW_HIDE);
if (bShowScroll)
m_hwndScrollBar.InvalidateRect(NULL, FALSE); // redraw
}
BOOL CAutoComplete::OnEditKeyDown(WPARAM wParam, LPARAM lParam)
{
TRACE("CAutoComplete::OnEditKeyDown(%p, %p)\n", this, wParam);
UINT vk = (UINT)wParam; // virtual key
switch (vk)
{
case VK_UP: case VK_DOWN:
case VK_PRIOR: case VK_NEXT:
// is suggestion available?
if (!CanAutoSuggest())
return FALSE; // do default
if (IsWindowVisible())
return OnListUpDown(vk);
break;
case VK_ESCAPE:
{
// is suggestion available?
if (!CanAutoSuggest())
return FALSE; // do default
if (IsWindowVisible())
{
SetEditText(m_strText); // revert the edit text
// select the end
INT cch = m_strText.GetLength();
SetEditSel(cch, cch);
HideDropDown(); // hide
return TRUE; // eat
}
break;
}
case VK_RETURN:
{
if (::GetKeyState(VK_CONTROL) < 0)
{
// quick edit
CStringW strText = GetEditText();
SetEditText(GetQuickEdit(strText));
}
else
{
// if item is selected, then update the edit text
INT iItem = m_hwndList.GetCurSel();
if (iItem != -1)
{
CStringW strText = GetItemText(iItem);
SetEditText(strText);
}
}
// select all
INT cch = ::CallWindowProcW(m_fnOldEditProc, m_hwndEdit, WM_GETTEXTLENGTH, 0, 0);
SetEditSel(0, cch);
// hide
HideDropDown();
break;
}
case VK_TAB:
{
// ACO_USETAB
if (IsWindowVisible() && UseTab())
{
if (GetKeyState(VK_SHIFT) < 0)
return OnListUpDown(VK_UP);
else
return OnListUpDown(VK_DOWN);
}
break;
}
case VK_DELETE:
{
// is suggestion available?
if (!CanAutoSuggest())
return FALSE; // do default
::DefSubclassProc(m_hwndEdit, WM_KEYDOWN, VK_DELETE, 0); // do default
UpdateCompletion(FALSE);
return TRUE; // eat
}
case VK_BACK:
{
if (::GetKeyState(VK_CONTROL) < 0)
{
DoBackWord();
return TRUE; // eat
}
break;
}
}
return FALSE; // default
}
LRESULT CAutoComplete::OnEditChar(WPARAM wParam, LPARAM lParam)
{
TRACE("CACEditCtrl::OnEditChar(%p, %p)\n", this, wParam);
if (wParam == L'\n' || wParam == L'\t')
return 0; // eat
LRESULT ret = ::DefSubclassProc(m_hwndEdit, WM_CHAR, wParam, lParam); // do default
if (CanAutoSuggest() || CanAutoAppend())
UpdateCompletion(wParam != VK_BACK);
return ret;
}
VOID CAutoComplete::OnListSelChange()
{
// update EDIT text
INT iItem = m_hwndList.GetCurSel();
CStringW text = ((iItem != -1) ? GetItemText(iItem) : m_strText);
SetEditText(text);
// ensure the item visible
m_hwndList.EnsureVisible(iItem, FALSE);
// select the end
INT cch = text.GetLength();
SetEditSel(cch, cch);
}
BOOL CAutoComplete::OnListUpDown(UINT vk)
{
if (!CanAutoSuggest())
return FALSE; // default
if ((m_dwOptions & ACO_UPDOWNKEYDROPSLIST) && !IsWindowVisible())
{
ShowDropDown();
return TRUE; // eat
}
INT iItem = m_hwndList.GetCurSel(); // current selection
INT cItems = m_hwndList.GetItemCount(); // the number of items
switch (vk)
{
case VK_UP:
if (iItem == -1)
iItem = cItems - 1;
else if (iItem == 0)
iItem = -1;
else
--iItem;
m_hwndList.SetCurSel(iItem);
break;
case VK_DOWN:
if (iItem == -1)
iItem = 0;
else if (iItem == cItems - 1)
iItem = -1;
else
++iItem;
m_hwndList.SetCurSel(iItem);
break;
case VK_PRIOR:
if (iItem == -1)
{
iItem = cItems - 1;
}
else if (iItem == 0)
{
iItem = -1;
}
else
{
iItem -= m_hwndList.GetVisibleCount() - 1;
if (iItem < 0)
iItem = 0;
}
m_hwndList.SetCurSel(iItem);
break;
case VK_NEXT:
if (iItem == -1)
{
iItem = 0;
}
else if (iItem == cItems - 1)
{
iItem = -1;
}
else
{
iItem += m_hwndList.GetVisibleCount() - 1;
if (iItem > cItems)
iItem = cItems - 1;
}
m_hwndList.SetCurSel(iItem);
break;
default:
{
ATLASSERT(FALSE);
break;
}
}
return TRUE; // eat
}
//////////////////////////////////////////////////////////////////////////////
// CAutoComplete IAutoComplete methods
// @implemented
STDMETHODIMP CAutoComplete::Enable(BOOL fEnable)
{
TRACE("(%p)->Enable(%d)\n", this, fEnable);
m_bEnabled = fEnable;
return S_OK;
}
STDMETHODIMP
CAutoComplete::Init(HWND hwndEdit, IUnknown *punkACL,
LPCOLESTR pwszRegKeyPath, LPCOLESTR pwszQuickComplete)
{
TRACE("(%p)->Init(0x%08lx, %p, %s, %s)\n",
this, hwndEdit, punkACL, debugstr_w(pwszRegKeyPath), debugstr_w(pwszQuickComplete));
// sanity check
if (m_hwndEdit || !punkACL)
return E_FAIL;
if (!hwndEdit)
return E_INVALIDARG;
// do subclass textbox to watch messages
m_fnOldEditProc = reinterpret_cast<WNDPROC>(::GetWindowLongPtrW(hwndEdit, GWLP_WNDPROC));
if (!m_fnOldEditProc)
return E_FAIL;
if (!::SetWindowSubclass(hwndEdit, EditSubclassProc, 0, reinterpret_cast<DWORD_PTR>(this)))
return E_FAIL;
m_hwndEdit = hwndEdit;
// add reference to m_hwndEdit
AddRef();
// set word break procedure
m_fnOldWordBreakProc = reinterpret_cast<EDITWORDBREAKPROCW>(
::SendMessageW(m_hwndEdit, EM_SETWORDBREAKPROC, 0,
reinterpret_cast<LPARAM>(EditWordBreakProcW)));
// save position
::GetWindowRect(m_hwndEdit, &m_rcEdit);
// get an IEnumString
ATLASSERT(!m_pEnum);
punkACL->QueryInterface(IID_IEnumString, (VOID **)&m_pEnum);
TRACE("m_pEnum: %p\n", static_cast<void *>(m_pEnum));
if (m_pEnum)
m_pEnum->AddRef(); // hold not to be freed
// get an IACList
ATLASSERT(!m_pACList);
punkACL->QueryInterface(IID_IACList, (VOID **)&m_pACList);
TRACE("m_pACList: %p\n", static_cast<void *>(m_pACList));
if (m_pACList)
m_pACList->AddRef(); // hold not to be freed
UpdateDropDownState(); // create/hide the drop-down window if necessary
// load quick completion info
LoadQuickComplete(pwszRegKeyPath, pwszQuickComplete);
// any combobox for m_hwndEdit?
m_hwndCombo = NULL;
HWND hwndParent = ::GetParent(m_hwndEdit);
WCHAR szClass[16];
if (::GetClassNameW(hwndParent, szClass, _countof(szClass)))
{
if (::StrCmpIW(szClass, WC_COMBOBOXW) == 0 ||
::StrCmpIW(szClass, WC_COMBOBOXEXW) == 0)
{
m_hwndCombo = hwndParent; // get combobox
}
}
return S_OK;
}
//////////////////////////////////////////////////////////////////////////////
// CAutoComplete IAutoComplete2 methods
// @implemented
STDMETHODIMP CAutoComplete::GetOptions(DWORD *pdwFlag)
{
TRACE("(%p) -> (%p)\n", this, pdwFlag);
if (pdwFlag)
{
*pdwFlag = m_dwOptions;
return S_OK;
}
return E_INVALIDARG;
}
// @implemented
STDMETHODIMP CAutoComplete::SetOptions(DWORD dwFlag)
{
TRACE("(%p) -> (0x%x)\n", this, dwFlag);
m_dwOptions = dwFlag;
if (m_dwOptions & ACO_SEARCH)
FIXME(" ACO_SEARCH not supported\n");
if (m_dwOptions & ACO_FILTERPREFIXES)
FIXME(" ACO_FILTERPREFIXES not supported\n");
if (m_dwOptions & ACO_RTLREADING)
FIXME(" ACO_RTLREADING not supported\n");
UpdateDropDownState(); // create/hide the drop-down window if necessary
return S_OK;
}
//////////////////////////////////////////////////////////////////////////////
// CAutoComplete IAutoCompleteDropDown methods
// @implemented
STDMETHODIMP CAutoComplete::GetDropDownStatus(DWORD *pdwFlags, LPWSTR *ppwszString)
{
BOOL dropped = m_hwndList.IsWindowVisible();
if (pdwFlags)
*pdwFlags = (dropped ? ACDD_VISIBLE : 0);
if (ppwszString)
{
*ppwszString = NULL;
if (dropped)
{
// get selected item
INT iItem = m_hwndList.GetCurSel();
if (iItem >= 0)
{
// get the text of item
CStringW strText = m_hwndList.GetItemText(iItem);
// store to *ppwszString
SHStrDupW(strText, ppwszString);
if (*ppwszString == NULL)
return E_OUTOFMEMORY;
}
}
}
return S_OK;
}
STDMETHODIMP CAutoComplete::ResetEnumerator()
{
FIXME("(%p): stub\n", this);
Reset();
m_innerList.RemoveAll();
return S_OK;
}
//////////////////////////////////////////////////////////////////////////////
// CAutoComplete IEnumString methods
// @implemented
STDMETHODIMP CAutoComplete::Next(ULONG celt, LPOLESTR *rgelt, ULONG *pceltFetched)
{
TRACE("(%p, %d, %p, %p)\n", this, celt, rgelt, pceltFetched);
if (rgelt)
*rgelt = NULL;
if (*pceltFetched)
*pceltFetched = 0;
if (celt != 1 || !rgelt || !pceltFetched || !m_pEnum)
return E_INVALIDARG;
LPWSTR pszText = NULL;
HRESULT hr = m_pEnum->Next(1, &pszText, pceltFetched);
if (hr == S_OK)
*rgelt = pszText;
else
::CoTaskMemFree(pszText);
return hr;
}
// @implemented
STDMETHODIMP CAutoComplete::Skip(ULONG celt)
{
TRACE("(%p, %d)\n", this, celt);
return E_NOTIMPL;
}
// @implemented
STDMETHODIMP CAutoComplete::Reset()
{
TRACE("(%p)\n", this);
if (m_pEnum)
return m_pEnum->Reset();
return E_FAIL;
}
// @implemented
STDMETHODIMP CAutoComplete::Clone(IEnumString **ppOut)
{
TRACE("(%p, %p)\n", this, ppOut);
if (ppOut)
*ppOut = NULL;
return E_NOTIMPL;
}
//////////////////////////////////////////////////////////////////////////////
// CAutoComplete protected methods
VOID CAutoComplete::UpdateDropDownState()
{
if (CanAutoSuggest())
{
// create the drop-down window if not existed
if (!m_hWnd)
CreateDropDown();
}
else
{
// hide if existed
if (m_hWnd)
ShowWindow(SW_HIDE);
}
}
// calculate the positions of the controls
VOID CAutoComplete::CalcRects(BOOL bDowner, RECT& rcList, RECT& rcScrollBar, RECT& rcSizeBox) const
{
// get the client rectangle
RECT rcClient;
GetClientRect(&rcClient);
// the list
rcList = rcClient;
rcList.right = rcList.left + CX_LIST;
// the scroll bar
rcScrollBar = rcClient;
rcScrollBar.left = rcClient.right - GetSystemMetrics(SM_CXVSCROLL);
if (bDowner)
{
rcScrollBar.top = 0;
rcScrollBar.bottom = rcClient.bottom - GetSystemMetrics(SM_CYHSCROLL);
}
else
{
rcScrollBar.top = GetSystemMetrics(SM_CYHSCROLL);
}
// the size box
rcSizeBox = rcClient;
rcSizeBox.left = rcClient.right - GetSystemMetrics(SM_CXVSCROLL);
if (bDowner)
{
rcSizeBox.top = rcClient.bottom - GetSystemMetrics(SM_CYHSCROLL);
}
else
{
rcSizeBox.top = 0;
rcSizeBox.bottom = rcClient.top + GetSystemMetrics(SM_CYHSCROLL);
}
}
VOID CAutoComplete::LoadQuickComplete(LPCWSTR pwszRegKeyPath, LPCWSTR pwszQuickComplete)
{
m_strQuickComplete.Empty();
if (pwszRegKeyPath)
{
CStringW strPath = pwszRegKeyPath;
INT ichSep = strPath.ReverseFind(L'\\'); // find separator
if (ichSep != -1) // found the last separator
{
// split by the separator
CStringW strKey = strPath.Left(ichSep);
CStringW strName = strPath.Mid(ichSep + 1);
// load from registry
WCHAR szValue[MAX_PATH] = L"";
DWORD cbValue = sizeof(szValue), dwType = REG_NONE;
SHRegGetUSValueW(pwszRegKeyPath, strName, &dwType,
szValue, &cbValue, FALSE, NULL, 0);
if (szValue[0] != 0 && cbValue != 0 &&
(dwType == REG_SZ || dwType == REG_EXPAND_SZ))
{
m_strQuickComplete = szValue;
}
}
}
if (pwszQuickComplete && m_strQuickComplete.IsEmpty())
{
m_strQuickComplete = pwszQuickComplete;
}
}
CStringW CAutoComplete::GetQuickEdit(LPCWSTR pszText) const
{
if (pszText[0] == 0 || m_strQuickComplete.IsEmpty())
return pszText;
// m_strQuickComplete will be "www.%s.com" etc.
CStringW ret;
ret.Format(m_strQuickComplete, pszText);
return ret;
}
VOID CAutoComplete::RepositionDropDown()
{
// get nearest monitor from m_hwndEdit
HMONITOR hMon = ::MonitorFromWindow(m_hwndEdit, MONITOR_DEFAULTTONEAREST);
ATLASSERT(hMon != NULL);
if (hMon == NULL)
return;
// get nearest monitor info
MONITORINFO mi = { sizeof(mi) };
if (!::GetMonitorInfo(hMon, &mi))
{
ATLASSERT(FALSE);
return;
}
// get count and item height
INT cItems = GetItemCount();
INT cyItem = m_hwndList.m_cyItem;
ATLASSERT(cyItem > 0);
// get m_hwndEdit position
RECT rcEdit;
::GetWindowRect(m_hwndEdit, &rcEdit);
INT x = rcEdit.left, y = rcEdit.bottom;
// get list extent
RECT rcMon = mi.rcMonitor;
INT cx = rcEdit.right - rcEdit.left, cy = cItems * cyItem;
BOOL bLongList = FALSE;
if (cy > CY_LIST)
{
cy = INT(CY_LIST / cyItem) * cyItem;
bLongList = TRUE;
}
// convert rectangle for frame
RECT rc = { 0, 0, cx, cy };
AdjustWindowRectEx(&rc, GetStyle(), FALSE, GetExStyle());
cy = rc.bottom - rc.top;
if (!m_bResized)
{
// is the drop-down window a 'downer' or 'upper'?
// NOTE: 'downer' is below the EDIT control. 'upper' is above the EDIT control.
m_bDowner = (rcEdit.bottom + cy < rcMon.bottom);
}
// adjust y and cy
if (m_bDowner)
{
if (rcMon.bottom < y + cy)
{
cy = ((rcMon.bottom - y) / cyItem) * cyItem;
bLongList = TRUE;
}
}
else
{
if (rcEdit.top < rcMon.top + cy)
{
cy = ((rcEdit.top - rcMon.top) / cyItem) * cyItem;
bLongList = TRUE;
}
y = rcEdit.top - cy;
}
// set status
m_hwndSizeBox.SetStatus(m_bDowner, bLongList);
if (m_bResized) // already resized?
PostMessageW(WM_SIZE, 0, 0); // re-layout
else
MoveWindow(x, y, cx, cy); // move
// show without activation
ShowWindow(SW_SHOWNOACTIVATE);
}
inline BOOL
CAutoComplete::DoesMatch(const CStringW& strTarget, const CStringW& strText) const
{
CStringW strBody;
if (DropPrefix(strTarget, strBody))
{
if (::StrCmpNIW(strBody, strText, strText.GetLength()) == 0)
return TRUE;
}
else if (::StrCmpNIW(strTarget, strText, strText.GetLength()) == 0)
{
return TRUE;
}
return FALSE;
}
VOID CAutoComplete::ScrapeOffList(const CStringW& strText, CSimpleArray<CStringW>& array)
{
for (INT iItem = array.GetSize() - 1; iItem >= 0; --iItem)
{
if (!DoesMatch(array[iItem], strText))
array.RemoveAt(iItem);
}
}
VOID CAutoComplete::ReLoadInnerList(const CStringW& strText)
{
m_innerList.RemoveAll(); // clear contents
m_bPartialList = FALSE;
if (!m_pEnum || strText.IsEmpty())
return;
// reload the items
LPWSTR pszItem;
ULONG cGot;
CStringW strTarget;
HRESULT hr;
for (;;)
{
// get next item
hr = m_pEnum->Next(1, &pszItem, &cGot);
//TRACE("m_pEnum->Next(%p): 0x%08lx\n", reinterpret_cast<IUnknown *>(m_pEnum), hr);
if (hr != S_OK)
break;
strTarget = pszItem;
::CoTaskMemFree(pszItem); // free
if (m_bPartialList) // if items are too many
{
// do filter the items
if (DoesMatch(strTarget, strText))
{
m_innerList.Add(strTarget);
if (m_innerList.GetSize() >= MAX_ITEM_COUNT)
break;
}
}
else
{
m_innerList.Add(strTarget); // append item to m_innerList
// if items are too many
if (m_innerList.GetSize() >= MAX_ITEM_COUNT)
{
// filter the items now
m_bPartialList = TRUE;
ScrapeOffList(strText, m_innerList);
if (m_innerList.GetSize() >= MAX_ITEM_COUNT)
break;
}
}
// check the timeout
if (::GetTickCount() - m_dwTick >= COMPLETION_TIMEOUT)
break; // too late
}
}
// update inner list and m_strText and m_strStemText
VOID CAutoComplete::UpdateInnerList(const CStringW& strText)
{
BOOL bReset = FALSE, bExpand = FALSE; // flags
// if previous text was empty
if (m_strText.IsEmpty())
{
bReset = TRUE;
}
// save text
m_strText = strText;
// do expand the items if the stem is changed
CStringW strStemText = GetStemText(strText);
if (m_strStemText.CompareNoCase(strStemText) != 0)
{
m_strStemText = strStemText;
bReset = TRUE;
bExpand = !m_strStemText.IsEmpty();
}
// if the previous enumeration is too large
if (m_bPartialList)
bReset = bExpand = TRUE; // retry enumeratation
// reset if necessary
if (bReset && m_pEnum)
{
HRESULT hr = m_pEnum->Reset(); // IEnumString::Reset
TRACE("m_pEnum->Reset(%p): 0x%08lx\n",
static_cast<IUnknown *>(m_pEnum), hr);
}
// update ac list if necessary
if (bExpand && m_pACList)
{
HRESULT hr = m_pACList->Expand(strStemText); // IACList::Expand
TRACE("m_pACList->Expand(%p, %S): 0x%08lx\n",
static_cast<IUnknown *>(m_pACList),
static_cast<LPCWSTR>(strStemText), hr);
}
if (bExpand || m_innerList.GetSize() == 0)
{
// reload the inner list
ReLoadInnerList(strText);
}
}
VOID CAutoComplete::UpdateOuterList(const CStringW& strText)
{
if (strText.IsEmpty())
{
m_outerList.RemoveAll();
return;
}
if (m_bPartialList)
{
// it is already filtered
m_outerList = m_innerList;
}
else
{
// do filtering
m_outerList.RemoveAll();
for (INT iItem = 0; iItem < m_innerList.GetSize(); ++iItem)
{
const CStringW& strTarget = m_innerList[iItem];
if (DoesMatch(strTarget, strText))
m_outerList.Add(strTarget);
// check the timeout
if (::GetTickCount() - m_dwTick >= COMPLETION_TIMEOUT)
break; // too late
}
}
if (::GetTickCount() - m_dwTick < COMPLETION_TIMEOUT)
{
// sort and unique
DoSort(m_outerList);
DoUniqueAndTrim(m_outerList);
}
// set the item count of the virtual listview
m_hwndList.SendMessageW(LVM_SETITEMCOUNT, m_outerList.GetSize(), 0);
}
VOID CAutoComplete::UpdateCompletion(BOOL bAppendOK)
{
TRACE("CAutoComplete::UpdateCompletion(%p, %d)\n", this, bAppendOK);
m_dwTick = GetTickCount(); // to check the timeout
CStringW strText = GetEditText();
if (m_strText.CompareNoCase(strText) == 0)
{
// no change
return;
}
// update inner list
UpdateInnerList(strText);
if (m_innerList.GetSize() <= 0) // no items
{
HideDropDown();
return;
}
if (CanAutoSuggest()) // can we auto-suggest?
{
m_bInSelectItem = TRUE; // don't respond
SelectItem(-1); // select none
m_bInSelectItem = FALSE;
UpdateOuterList(strText);
if (m_outerList.GetSize() > 0)
RepositionDropDown();
else
HideDropDown();
}
if (CanAutoAppend() && bAppendOK) // can we auto-append?
{
DoAutoAppend();
}
}
//////////////////////////////////////////////////////////////////////////////
// CAutoComplete message handlers
// WM_CREATE
// This message is sent when the window is about to be created after WM_NCCREATE.
// The return value is -1 (failure) or zero (success).
LRESULT CAutoComplete::OnCreate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
TRACE("CAutoComplete::OnCreate(%p)\n", this);
// set the pointer of CAutoComplete
m_hwndList.m_pDropDown = this;
m_hwndScrollBar.m_pDropDown = this;
m_hwndSizeBox.m_pDropDown = this;
// create the children
m_hwndList.Create(m_hWnd);
if (!m_hwndList)
return -1; // failure
m_hwndSizeBox.Create(m_hWnd);
if (!m_hwndSizeBox)
return -1; // failure
m_hwndScrollBar.Create(m_hWnd);
if (!m_hwndScrollBar)
return -1; // failure
// show the controls
m_hwndList.ShowWindow(SW_SHOWNOACTIVATE);
m_hwndSizeBox.ShowWindow(SW_SHOWNOACTIVATE);
m_hwndScrollBar.ShowWindow(SW_SHOWNOACTIVATE);
// set the list font
m_hFont = reinterpret_cast<HFONT>(::GetStockObject(DEFAULT_GUI_FONT));
m_hwndList.SetFont(m_hFont);
// add reference to CAutoComplete::m_hWnd
AddRef();
return 0; // success
}
// WM_NCDESTROY
LRESULT CAutoComplete::OnNCDestroy(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
TRACE("CAutoComplete::OnNCDestroy(%p)\n", this);
// hide
if (IsWindowVisible())
HideDropDown();
// clear CAutoComplete pointers
m_hwndList.m_pDropDown = NULL;
m_hwndScrollBar.m_pDropDown = NULL;
m_hwndSizeBox.m_pDropDown = NULL;
// destroy controls
m_hwndList.DestroyWindow();
m_hwndScrollBar.DestroyWindow();
m_hwndSizeBox.DestroyWindow();
// clean up
m_hwndCombo = NULL;
// remove reference to CAutoComplete::m_hWnd
Release();
return 0;
}
// WM_EXITSIZEMOVE
// This message is sent once to a window after it has exited the moving or sizing mode.
LRESULT CAutoComplete::OnExitSizeMove(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
TRACE("CAutoComplete::OnExitSizeMove(%p)\n", this);
m_bResized = TRUE; // remember resized
ModifyStyle(WS_THICKFRAME, 0); // remove thick frame to resize
// frame changed
UINT uSWP_ = SWP_NOMOVE | SWP_NOSIZE | SWP_NOZORDER | SWP_FRAMECHANGED | SWP_NOACTIVATE;
SetWindowPos(NULL, 0, 0, 0, 0, uSWP_);
::SetFocus(m_hwndEdit); // restore focus
return 0;
}
// WM_DRAWITEM @implemented
// This message is sent to the owner window to draw m_hwndList.
LRESULT CAutoComplete::OnDrawItem(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
LPDRAWITEMSTRUCT pDraw = reinterpret_cast<LPDRAWITEMSTRUCT>(lParam);
ATLASSERT(pDraw != NULL);
ATLASSERT(m_hwndList.GetStyle() & LVS_OWNERDRAWFIXED);
// sanity check
if (pDraw->CtlType != ODT_LISTVIEW || pDraw->hwndItem != m_hwndList)
return FALSE;
// item rectangle
RECT rcItem = pDraw->rcItem;
// get info
UINT iItem = pDraw->itemID; // the index of item
CStringW strItem = m_hwndList.GetItemText(iItem); // get text of item
// draw background and set text color
HDC hDC = pDraw->hDC;
BOOL bSelected = (pDraw->itemState & ODS_SELECTED);
if (bSelected)
{
::FillRect(hDC, &rcItem, ::GetSysColorBrush(COLOR_HIGHLIGHT));
::SetTextColor(hDC, ::GetSysColor(COLOR_HIGHLIGHTTEXT));
}
else
{
::FillRect(hDC, &rcItem, ::GetSysColorBrush(COLOR_WINDOW));
::SetTextColor(hDC, ::GetSysColor(COLOR_WINDOWTEXT));
}
// draw text
rcItem.left += ::GetSystemMetrics(SM_CXBORDER);
HGDIOBJ hFontOld = ::SelectObject(hDC, m_hFont);
const UINT uDT_ = DT_LEFT | DT_NOPREFIX | DT_SINGLELINE | DT_VCENTER;
::SetBkMode(hDC, TRANSPARENT);
::DrawTextW(hDC, strItem, -1, &rcItem, uDT_);
::SelectObject(hDC, hFontOld);
return TRUE;
}
// WM_GETMINMAXINFO @implemented
// This message is sent to a window when the size or position of the window is about to change.
LRESULT CAutoComplete::OnGetMinMaxInfo(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
// restrict minimum size
LPMINMAXINFO pInfo = reinterpret_cast<LPMINMAXINFO>(lParam);
pInfo->ptMinTrackSize.x = ::GetSystemMetrics(SM_CXVSCROLL);
pInfo->ptMinTrackSize.y = ::GetSystemMetrics(SM_CYHSCROLL);
return 0;
}
// WM_MEASUREITEM @implemented
// This message is sent to the owner window to get the item extent of m_hwndList.
LRESULT CAutoComplete::OnMeasureItem(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
LPMEASUREITEMSTRUCT pMeasure = reinterpret_cast<LPMEASUREITEMSTRUCT>(lParam);
ATLASSERT(pMeasure != NULL);
if (pMeasure->CtlType != ODT_LISTVIEW)
return FALSE;
if (!m_hwndList)
return FALSE;
ATLASSERT(m_hwndList.GetStyle() & LVS_OWNERDRAWFIXED);
pMeasure->itemHeight = m_hwndList.m_cyItem; // height of item
return TRUE;
}
// WM_MOUSEACTIVATE @implemented
// The return value of this message specifies whether the window should be activated or not.
LRESULT CAutoComplete::OnMouseActivate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
return MA_NOACTIVATE; // don't activate by mouse
}
// WM_NCACTIVATE
// This message is sent to a window to indicate an active or inactive state.
LRESULT CAutoComplete::OnNCActivate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
bHandled = FALSE; // do default
return 0;
}
// WM_NCLBUTTONDOWN
LRESULT CAutoComplete::OnNCLButtonDown(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
switch (wParam)
{
case HTBOTTOMRIGHT: case HTTOPRIGHT:
{
// add thick frame to resize.
ModifyStyle(0, WS_THICKFRAME);
// frame changed
UINT uSWP_ = SWP_NOMOVE | SWP_NOSIZE | SWP_NOZORDER | SWP_FRAMECHANGED | SWP_NOACTIVATE;
SetWindowPos(NULL, 0, 0, 0, 0, uSWP_);
break;
}
}
bHandled = FALSE; // do default
return 0;
}
// WM_NOTIFY
// This message informs the parent window of a control that an event has occurred.
LRESULT CAutoComplete::OnNotify(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
LPNMHDR pnmh = reinterpret_cast<LPNMHDR>(lParam);
ATLASSERT(pnmh != NULL);
switch (pnmh->code)
{
case NM_DBLCLK: // double-clicked
{
TRACE("NM_DBLCLK\n");
HideDropDown();
break;
}
case NM_HOVER: // mouse is hovering
{
POINT pt;
::GetCursorPos(&pt); // get cursor position in screen coordinates
m_hwndList.ScreenToClient(&pt); // into client coordinates
INT iItem = m_hwndList.ItemFromPoint(pt.x, pt.y);
if (iItem != -1)
{
m_bInSelectItem = TRUE; // don't respond
m_hwndList.SetCurSel(iItem); // select
m_bInSelectItem = FALSE;
}
return TRUE; // eat
}
case LVN_GETDISPINFOA: // for user's information only
{
TRACE("LVN_GETDISPINFOA\n");
if (pnmh->hwndFrom != m_hwndList)
break;
LV_DISPINFOA *pDispInfo = reinterpret_cast<LV_DISPINFOA *>(pnmh);
LV_ITEMA *pItem = &pDispInfo->item;
INT iItem = pItem->iItem;
if (iItem == -1)
break;
CStringW strText = GetItemText(iItem);
if (pItem->mask & LVIF_TEXT)
SHUnicodeToAnsi(strText, pItem->pszText, pItem->cchTextMax);
break;
}
case LVN_GETDISPINFOW: // for user's information only
{
TRACE("LVN_GETDISPINFOW\n");
if (pnmh->hwndFrom != m_hwndList)
break;
LV_DISPINFOW *pDispInfo = reinterpret_cast<LV_DISPINFOW *>(pnmh);
LV_ITEMW *pItem = &pDispInfo->item;
INT iItem = pItem->iItem;
if (iItem == -1)
break;
CStringW strText = GetItemText(iItem);
if (pItem->mask & LVIF_TEXT)
StringCbCopyW(pItem->pszText, pItem->cchTextMax, strText);
break;
}
case LVN_HOTTRACK: // enabled by LVS_EX_TRACKSELECT
{
TRACE("LVN_HOTTRACK\n");
LPNMLISTVIEW pListView = reinterpret_cast<LPNMLISTVIEW>(pnmh);
INT iItem = pListView->iItem;
TRACE("LVN_HOTTRACK: iItem:%d\n", iItem);
m_hwndList.SetCurSel(iItem);
m_hwndList.EnsureVisible(iItem, FALSE);
return TRUE;
}
case LVN_ITEMACTIVATE: // enabled by LVS_EX_ONECLICKACTIVATE
{
TRACE("LVN_ITEMACTIVATE\n");
LPNMITEMACTIVATE pItemActivate = reinterpret_cast<LPNMITEMACTIVATE>(pnmh);
INT iItem = pItemActivate->iItem;
TRACE("LVN_ITEMACTIVATE: iItem:%d\n", iItem);
if (iItem != -1) // the item is clicked
{
SelectItem(iItem);
HideDropDown();
}
break;
}
case LVN_ITEMCHANGED: // item info is changed
{
TRACE("LVN_ITEMCHANGED\n");
LPNMLISTVIEW pListView = reinterpret_cast<LPNMLISTVIEW>(pnmh);
if (pListView->uChanged & LVIF_STATE) // selection changed
{
// listview selection changed
if (!m_bInSelectItem)
{
OnListSelChange();
}
UpdateScrollBar();
}
break;
}
}
return 0;
}
// WM_NCHITTEST @implemented
// The return value is indicating the cursor shape and the behaviour.
LRESULT CAutoComplete::OnNCHitTest(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
TRACE("CAutoComplete::OnNCHitTest(%p)\n", this);
POINT pt = { GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam) }; // in screen coordinates
ScreenToClient(&pt); // into client coordinates
if (ChildWindowFromPoint(pt) == m_hwndSizeBox) // hit?
{
// allow resizing (with cursor shape)
return m_bDowner ? HTBOTTOMRIGHT : HTTOPRIGHT;
}
bHandled = FALSE; // do default
return 0;
}
// WM_SIZE @implemented
// This message is sent when the window size is changed.
LRESULT CAutoComplete::OnSize(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
// calculate the positions of the controls
CRect rcList, rcScrollBar, rcSizeBox;
CalcRects(m_bDowner, rcList, rcScrollBar, rcSizeBox);
// reposition the controls in smartest way
UINT uSWP_ = SWP_NOACTIVATE | SWP_NOCOPYBITS;
HDWP hDWP = ::BeginDeferWindowPos(3);
hDWP = ::DeferWindowPos(hDWP, m_hwndScrollBar, HWND_TOP,
rcScrollBar.left, rcScrollBar.top,
rcScrollBar.Width(), rcScrollBar.Height(), uSWP_);
hDWP = ::DeferWindowPos(hDWP, m_hwndSizeBox, m_hwndScrollBar,
rcSizeBox.left, rcSizeBox.top,
rcSizeBox.Width(), rcSizeBox.Height(), uSWP_);
hDWP = ::DeferWindowPos(hDWP, m_hwndList, m_hwndSizeBox,
rcList.left, rcList.top,
rcList.Width(), rcList.Height(), uSWP_);
::EndDeferWindowPos(hDWP);
UpdateScrollBar();
return 0;
}
// WM_SHOWWINDOW
LRESULT CAutoComplete::OnShowWindow(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
// hook mouse events
BOOL bShow = (BOOL)wParam;
if (bShow)
{
if (s_hWatchWnd != m_hWnd && ::IsWindowVisible(s_hWatchWnd))
::ShowWindowAsync(s_hWatchWnd, SW_HIDE);
s_hWatchWnd = m_hWnd; // watch this
// unhook mouse if any
if (s_hMouseHook)
{
HHOOK hHookOld = s_hMouseHook;
s_hMouseHook = NULL;
::UnhookWindowsHookEx(hHookOld);
}
// hook mouse
s_hMouseHook = ::SetWindowsHookEx(WH_MOUSE, MouseProc, NULL, ::GetCurrentThreadId());
ATLASSERT(s_hMouseHook != NULL);
// set timer
SetTimer(WATCH_TIMER_ID, WATCH_INTERVAL, NULL);
bHandled = FALSE; // do default
return 0;
}
else
{
// kill timer
KillTimer(WATCH_TIMER_ID);
s_hWatchWnd = NULL; // unwatch
// unhook mouse if any
if (s_hMouseHook)
{
HHOOK hHookOld = s_hMouseHook;
s_hMouseHook = NULL;
::UnhookWindowsHookEx(hHookOld);
}
LRESULT ret = DefWindowProcW(uMsg, wParam, lParam); // do default
if (m_hwndCombo)
::InvalidateRect(m_hwndCombo, NULL, TRUE); // redraw
m_outerList.RemoveAll(); // no use
return ret;
}
}
// WM_TIMER
LRESULT CAutoComplete::OnTimer(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
if (wParam != WATCH_TIMER_ID) // sanity check
return 0;
// if the textbox is dead, then kill the timer
if (!::IsWindow(m_hwndEdit))
{
KillTimer(WATCH_TIMER_ID);
return 0;
}
// m_hwndEdit is moved?
RECT rcEdit;
::GetWindowRect(m_hwndEdit, &rcEdit);
if (!::EqualRect(&rcEdit, &m_rcEdit))
{
// if so, hide
HideDropDown();
m_rcEdit = rcEdit; // update rectangle
m_bResized = FALSE; // clear flag
}
return 0;
}
// WM_VSCROLL
// This message is sent when a scroll event occurs.
LRESULT CAutoComplete::OnVScroll(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
TRACE("CAutoComplete::OnVScroll(%p)\n", this);
WORD code = LOWORD(wParam);
switch (code)
{
case SB_THUMBPOSITION: case SB_THUMBTRACK:
{
// get the scrolling info
INT nPos = HIWORD(wParam);
SCROLLINFO si = { sizeof(si), SIF_ALL };
m_hwndList.GetScrollInfo(SB_VERT, &si);
// scroll the list-view by CListView::EnsureVisible
INT cItems = m_hwndList.GetItemCount();
// iItem : cItems == (nPos - si.nMin) : (si.nMax - si.nMin).
INT iItem = cItems * (nPos - si.nMin) / (si.nMax - si.nMin);
if (nPos > si.nPos)
{
iItem += m_hwndList.GetVisibleCount();
if (iItem >= cItems)
iItem = cItems - 1;
}
m_hwndList.EnsureVisible(iItem, FALSE);
// update scrolling position of m_hwndScrollBar
si.fMask = SIF_POS;
m_hwndList.GetScrollInfo(SB_VERT, &si);
m_hwndScrollBar.SetScrollInfo(SB_VERT, &si, FALSE);
break;
}
default:
{
// pass it to m_hwndList
m_hwndList.SendMessageW(WM_VSCROLL, wParam, lParam);
UpdateScrollBar();
break;
}
}
return 0;
}