#include "ctlspriv.h" #include "listview.h" #include "image.h" #include #include #include "uxthemep.h" #define __IOleControl_INTERFACE_DEFINED__ // There is a conflict with the IOleControl's def of CONTROLINFO #include "shlobj.h" #ifdef FULL_DEBUG #define LISTVIEW_VFX_DEFAULT TRUE #else #define LISTVIEW_VFX_DEFAULT FALSE #endif int LV_GetNewColWidth(LV* plv, int iFirst, int iLast); void ListView_RecalcTileSize(LV* plv); int ListView_ComputeCXItemSize(LV* plv); #define IE_SETTINGS TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Explorer\\Advanced") #define USE_DBL_CLICK_TIMER TEXT("UseDoubleClickTimer") int g_bUseDblClickTimer; #define LVMP_WINDOWPOSCHANGED (WM_USER + 1) HRESULT WINAPI UninitializeFlatSB(HWND hwnd); // the insert mark is 6 pixels wide #define INSERTMARKSIZE 6 #define GROUPHEADER_PADDING 6 #define GRADIENT_WIDTH 300 #define COLORISLIGHT(clr) ((5*GetGValue((clr)) + 2*GetRValue((clr)) + GetBValue((clr))) > 8*128) void ListView_HandleMouse(LV* plv, BOOL fDoubleClick, int x, int y, UINT keyFlags, BOOL bMouseWheel); /// function table setup const PFNLISTVIEW_DRAWITEM pfnListView_DrawItem[5] = { ListView_IDrawItem, ListView_RDrawItem, ListView_IDrawItem, ListView_LDrawItem, ListView_TDrawItem, }; void ListView_HandleStateIconClick(LV* plv, int iItem); DWORD ListView_IApproximateViewRect(LV* ,int, int, int); DWORD ListView_RApproximateViewRect(LV* ,int, int, int); DWORD ListView_LApproximateViewRect(LV* ,int, int, int); const PFNLISTVIEW_APPROXIMATEVIEWRECT pfnListView_ApproximateViewRect[5] = { ListView_IApproximateViewRect, ListView_RApproximateViewRect, ListView_IApproximateViewRect, ListView_LApproximateViewRect, ListView_IApproximateViewRect, }; const PFNLISTVIEW_UPDATESCROLLBARS pfnListView_UpdateScrollBars[5] = { ListView_IUpdateScrollBars, ListView_RUpdateScrollBars, ListView_IUpdateScrollBars, ListView_LUpdateScrollBars, ListView_IUpdateScrollBars, }; const PFNLISTVIEW_ITEMHITTEST pfnListView_ItemHitTest[5] = { ListView_IItemHitTest, ListView_RItemHitTest, ListView_SItemHitTest, ListView_LItemHitTest, ListView_TItemHitTest, }; const PFNLISTVIEW_ONSCROLL pfnListView_OnScroll[5] = { ListView_IOnScroll, ListView_ROnScroll, ListView_IOnScroll, ListView_LOnScroll, ListView_IOnScroll, }; const PFNLISTVIEW_SCROLL2 pfnListView_Scroll2[5] = { ListView_IScroll2, ListView_RScroll2, ListView_IScroll2, ListView_LScroll2, ListView_IScroll2, }; const PFNLISTVIEW_GETSCROLLUNITSPERLINE pfnListView_GetScrollUnitsPerLine[5] = { ListView_IGetScrollUnitsPerLine, ListView_RGetScrollUnitsPerLine, ListView_IGetScrollUnitsPerLine, ListView_LGetScrollUnitsPerLine, ListView_IGetScrollUnitsPerLine, }; void ListView_NULLRecomputeLabelSize(LV* plv, LISTITEM* pitem, int i, HDC hdc, BOOL fusepitem) { // Report and List view don't need a recompute } const PFNLISTVIEW_RECOMPUTELABELSIZE pfnListView_RecomputeLabelSize[5] = { ListView_IRecomputeLabelSize, ListView_IRecomputeLabelSize, ListView_IRecomputeLabelSize, ListView_IRecomputeLabelSize, ListView_TRecomputeLabelSize, }; BOOL ListView_NULLRecomputeEx(LV* plv, HDPA hdpaSort, int iFrom, BOOL fForce) { return FALSE; } const PFNLISTVIEW_RECOMPUTEEX pfnListView_RecomputeEx[5] = { ListView_IRecomputeEx, ListView_RRecomputeEx, ListView_IRecomputeEx, ListView_NULLRecomputeEx, ListView_IRecomputeEx, }; #ifdef DEBUG_PAINT void ListView_DebugDrawInvalidRegion(LV* plv, RECT* prc, HRGN hrgn) { HDC hdc; HBRUSH hbrush; int bkMode; static int s_iclr; static COLORREF s_aclr[] = { RGB(255, 0, 0), RGB(0, 255, 0), RGB(255, 255, 0), RGB(0, 255, 255), }; s_iclr = (s_iclr + 1) % ARRAYSIZE(s_aclr); hdc = GetDC(plv->ci.hwnd); hbrush = CreateHatchBrush(HS_DIAGCROSS, s_aclr[s_iclr]); bkMode = SetBkMode(hdc, TRANSPARENT); if (prc) { FillRect(hdc, prc, hbrush); } else if (hrgn) { FillRgn(hdc, hrgn, hbrush); } DeleteObject((HGDIOBJ)hbrush); SetBkMode(hdc, bkMode); ReleaseDC(plv->ci.hwnd, hdc); Sleep(120); } BOOL ListView_DebugDrawInvalidItem(LV* plv, int iItem) { RECT rcLabel; RECT rcIcon; ListView_GetRects(plv, iItem, QUERY_DEFAULT, &rcIcon, &rcLabel, NULL, NULL); ListView_DebugDrawInvalidRegion(plv, &rcIcon, NULL); ListView_DebugDrawInvalidRegion(plv, &rcLabel, NULL); return TRUE; } void ListView_DebugDisplayClipRegion(LV* plv, RECT* prc, HRGN hrgn) { HDC hdc = GetDC(plv->ci.hwnd); if (prc) { InvertRect(hdc, prc); } else if (hrgn) { InvertRgn(hdc, hrgn); } Sleep(120); if (prc) { InvertRect(hdc, prc); } else if (hrgn) { InvertRgn(hdc, hrgn); } ReleaseDC(plv->ci.hwnd, hdc); } #else #define ListView_DebugDrawInvalidItem(plv, iItem) FALSE #endif // redefine to trace at most calls to ListView_SendChange #define DM_LVSENDCHANGE 0 // penwin.h is messed up; define local stuff for now #define HN_BEGINDIALOG 40 // Lens/EditText/garbage detection dialog is about // to come up on this hedit/bedit #define HN_ENDDIALOG 41 // Lens/EditText/garbage detection dialog has // just been destroyed //--------------------------------------------------------- // no way am I gonna make TWO function calls where I can do FOUR comparisons! // #define RECTS_IN_SIZE(sz, r2) (!RECTS_NOT_IN_SIZE(sz, r2)) #define RECTS_NOT_IN_SIZE(sz, r2) (\ ((sz).cx <= (r2).left) ||\ (0 >= (r2).right) ||\ ((sz).cy <= (r2).top) ||\ (0 >= (r2).bottom)) //--------------------------------------------------------- void ListView_OnUpdate(LV* plv, int i); void ListView_OnDestroy(LV* plv); BOOL ListView_ValidateScrollParams(LV* plv, int * dx, int *dy); void ListView_ButtonSelect(LV* plv, int iItem, UINT keyFlags, BOOL bSelected); void ListView_DeselectAll(LV* plv, int iDontDeselect); void ListView_LRInvalidateBelow(LV* plv, int i, int fSmoothScroll); void ListView_IInvalidateBelow(LV* plv, int i); void ListView_InvalidateFoldedItem(LV* plv, int iItem, BOOL fSelectionOnly, UINT fRedraw); void ListView_ReleaseBkImage(LV *plv); void ListView_RecalcRegion(LV *plv, BOOL fForce, BOOL fRedraw); BOOL g_fSlowMachine = -1; BOOL ListView_Init(HINSTANCE hinst) { WNDCLASS wc; wc.lpfnWndProc = ListView_WndProc; wc.hCursor = LoadCursor(NULL, IDC_ARROW); wc.hIcon = NULL; wc.lpszMenuName = NULL; wc.hInstance = hinst; wc.lpszClassName = c_szListViewClass; wc.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1); // NULL; wc.style = CS_DBLCLKS | CS_GLOBALCLASS; wc.cbWndExtra = sizeof(LV*); wc.cbClsExtra = 0; if (!RegisterClass(&wc) && !GetClassInfo(hinst, c_szListViewClass, &wc)) return FALSE; return TRUE; } // Cancel tracking tooltips which are activated by item focus via keyboard void ListView_CancelTipTrack(LV* plv) { // Make sure in tracking mode if (plv->hwndToolTips) { // Cancel any pending timer KillTimer(plv->ci.hwnd, IDT_TRACKINGTIP); if (ListView_IsKbdTipTracking(plv)) { TOOLINFO ti = {0}; // Mark as tracking nothing plv->iTracking = LVKTT_NOTRACK; // Reset tooltip to non-tracking ti.cbSize = sizeof(TOOLINFO); ti.hwnd = plv->ci.hwnd; SendMessage(plv->hwndToolTips, TTM_GETTOOLINFO, 0, (LPARAM)&ti); SendMessage(plv->hwndToolTips, TTM_TRACKACTIVATE, FALSE, (LPARAM)&ti); // Switch tooltip window back to non-tracking (manual) mode ti.uFlags &= ~TTF_TRACK; SendMessage(plv->hwndToolTips, TTM_SETTOOLINFO, 0, (LPARAM)&ti); } } } BOOL ListView_GetRegIASetting(BOOL *pb) { HKEY hkey; BOOL bRet = FALSE; BOOL bValue = TRUE; if (RegOpenKeyEx(HKEY_CURRENT_USER, IE_SETTINGS, 0, KEY_READ, &hkey) == ERROR_SUCCESS) { DWORD dwType; DWORD dwValue; DWORD cbValue = sizeof(DWORD); if (RegQueryValueEx(hkey, (LPTSTR)USE_DBL_CLICK_TIMER, 0, &dwType, (LPBYTE)&dwValue, &cbValue) == ERROR_SUCCESS) { bValue = (BOOL)dwValue; bRet = TRUE; } RegCloseKey(hkey); } *pb = bValue; return bRet; } BOOL ListView_NotifyCacheHint(LV* plv, int iFrom, int iTo) { NM_CACHEHINT nm; if (iFrom <= iTo) { nm.iFrom = iFrom; nm.iTo = iTo; return !(BOOL)CCSendNotify(&plv->ci, LVN_ODCACHEHINT, &nm.hdr); } return FALSE; } void ListView_LazyCreateObjects(LV *plv, int iMin, int iMax) { for (; iMin < iMax; iMin++) NotifyWinEvent(EVENT_OBJECT_CREATE, plv->ci.hwnd, OBJID_CLIENT, 1 + iMin); } // // Owner-data causes MSAA lots of grief, because there is no way to tell // MSAA "I just created 25 million items". You have to tell it one at a // time. Instead of sending out 25 million "add item" notifications, we // just send them out as they scroll into view. // // plv->iMSAAMin and plv->iMSAAMax are the range of items we most // recently told MSAA about. MSAAMax is *exclusive*, just like RECTs. // It makes the math easier. // // We use iMSAAMin and iMSAAMax to avoid sending blatantly redundant // notifications, which would other happen very frequently. // void ListView_LazyCreateWinEvents(LV *plv, int iFrom, int iTo) { int iMin = iFrom; int iMax = iTo+1; // Convert from [From,To] to [Min,Max) // // If the incoming range is entirely contained within the existing // range, then there is nothing to do. This happens a lot. // if (iMin >= plv->iMSAAMin && iMax <= plv->iMSAAMax) return; // // If the incoming range is adjacent to or overlaps the low end // of the existing range... (This happens when scrolling backwards.) // if (iMin <= plv->iMSAAMin && iMax >= plv->iMSAAMin) { // Notify the low end. ListView_LazyCreateObjects(plv, iMin, plv->iMSAAMin); // Extend the list of things we've notified. plv->iMSAAMin = iMin; // Remove it from the things left to be notified. iMin = plv->iMSAAMax; } // // Now do the same thing to the top end. // (This happens when scrolling forwards.) // if (iMax >= plv->iMSAAMax && iMin <= plv->iMSAAMax) { // Notify the top end. ListView_LazyCreateObjects(plv, plv->iMSAAMax, iMax); // Extend the list of things we've notified. plv->iMSAAMax = iMax; // Remove it from the things left to be notified. iMax = plv->iMSAAMin; } // // If there are still things to be notified, then it means that the // incoming range isn't contiguous with the previous range, so throw // away the old range and just set it to the current range. // (This happens when you grab the scrollbar and jump to a completely // unrelated part of the listview.) // if (iMin < iMax) { plv->iMSAAMin = iMin; plv->iMSAAMax = iMax; ListView_LazyCreateObjects(plv, iMin, iMax); } } LRESULT ListView_RequestFindItem(LV* plv, CONST LV_FINDINFO* plvfi, int iStart) { NM_FINDITEM nm; nm.lvfi = *plvfi; nm.iStart = iStart; return CCSendNotify(&plv->ci, LVN_ODFINDITEM, &nm.hdr); } BOOL ListView_SendChange(LV* plv, int i, int iSubItem, int code, UINT oldState, UINT newState, UINT changed, LPARAM lParam) { NM_LISTVIEW nm; nm.iItem = i; nm.iSubItem = iSubItem; nm.uNewState = newState; nm.uOldState = oldState; nm.uChanged = changed; nm.ptAction.x = 0; nm.ptAction.y = 0; nm.lParam = lParam; return !CCSendNotify(&plv->ci, code, &nm.hdr); } void ListView_SendODChangeAndInvalidate(LV* plv, int iFrom, int iTo, UINT oldState, UINT newState) { NM_ODSTATECHANGE nm; nm.iFrom = iFrom; nm.iTo = iTo; nm.uNewState = newState; nm.uOldState = oldState; CCSendNotify(&plv->ci, LVN_ODSTATECHANGED, &nm.hdr); // Tell accessibility, "Selection changed in a complex way" NotifyWinEvent(EVENT_OBJECT_SELECTIONWITHIN, plv->ci.hwnd, OBJID_CLIENT, CHILDID_SELF); // considerable speed increase less than 100 to do this method // while over 100, the other method works faster if ((iTo - iFrom) > 100) { InvalidateRect(plv->ci.hwnd, NULL, FALSE); } else { while (iFrom <= iTo) { ListView_InvalidateItem(plv, iFrom, TRUE, RDW_INVALIDATE); iFrom++; } } } // // This function autoarranges or snaps to grid based on the style and ExStyle // // Note: AutoArrange overrides the snap-to-grid style. // void ListView_ArrangeOrSnapToGrid(LV *plv) { if (plv->ci.style & LVS_AUTOARRANGE) ListView_OnArrange(plv, LVA_DEFAULT); else if (plv->exStyle & LVS_EX_SNAPTOGRID) ListView_OnArrange(plv, LVA_SNAPTOGRID); } BOOL ListView_Notify(LV* plv, int i, int iSubItem, int code) { NM_LISTVIEW nm; nm.iItem = i; nm.iSubItem = iSubItem; nm.uNewState = nm.uOldState = 0; nm.uChanged = 0; nm.lParam = 0; if (!ListView_IsOwnerData(plv)) { if (code == LVN_DELETEITEM) { LISTITEM * pItem = ListView_GetItemPtr(plv, i); if (pItem) nm.lParam = pItem->lParam; } } return (BOOL)CCSendNotify(&plv->ci, code, &nm.hdr); } BOOL ListView_GetEmptyText(LV* plv) { NMLVDISPINFO nm; BOOL ret; TCHAR szText[80]; if (plv->fNoEmptyText) return FALSE; if (plv->pszEmptyText) return TRUE; // For each listview control, we will only send this notify // once if necessary. memset(&nm, 0, sizeof(NMLVDISPINFO)); nm.item.mask = LVIF_TEXT; nm.item.cchTextMax = ARRAYSIZE(szText); nm.item.pszText = szText; szText[0] = TEXT('\0'); ret = (BOOL)CCSendNotify(&plv->ci, LVN_GETEMPTYTEXT, &nm.hdr); if (ret) // save the text so we don't notify again. Str_Set(&plv->pszEmptyText, szText); else // set a flag so we don't notify again. plv->fNoEmptyText = TRUE; return ret; } void ListView_NotifyFocusEvent(LV *plv) { if (plv->iFocus != -1 && IsWindowVisible(plv->ci.hwnd) && GetFocus() == plv->ci.hwnd) NotifyWinEvent(EVENT_OBJECT_FOCUS, plv->ci.hwnd, OBJID_CLIENT, plv->iFocus+1); } // // Call this function when the listview has changed in a radical manner. // It notifies MSAA that "Whoa, things are completely different now." // void ListView_NotifyRecreate(LV *plv) { NotifyWinEvent(EVENT_OBJECT_DESTROY, plv->ci.hwnd, OBJID_CLIENT, CHILDID_SELF); NotifyWinEvent(EVENT_OBJECT_CREATE, plv->ci.hwnd, OBJID_CLIENT, CHILDID_SELF); plv->iMSAAMin = plv->iMSAAMax = 0; } int ListView_OnSetItemCount(LV *plv, int iItems, DWORD dwFlags) { BOOL frt = TRUE; // For compatability we assume 0 for flags implies old (Athena) type of functionality and // does a Invalidate all otherwise if low bit is set we try to be a bit smarter. First pass // If the first added item is visible invalidate all. Yes we can do better... if (ListView_IsOwnerData(plv)) { int iItem; int cTotalItemsOld = plv->cTotalItems; BOOL fInvalidateAll = ((dwFlags & LVSICF_NOINVALIDATEALL) == 0); if ((iItems >= 0) && (iItems <= MAX_LISTVIEWITEMS)) { plv->cTotalItems = iItems; // check focus if (plv->iFocus >= iItems) plv->iFocus = -1; if (plv->iDropHilite >= iItems) plv->iDropHilite = -1; // check mark if (plv->iMark >= iItems) plv->iMark = -1; // make sure no selections above number of items plv->plvrangeCut->lpVtbl->ExcludeRange(plv->plvrangeCut, iItems, SELRANGE_MAXVALUE); if (FAILED(plv->plvrangeSel->lpVtbl->ExcludeRange(plv->plvrangeSel, iItems, SELRANGE_MAXVALUE))) { SetLastError(ERROR_OUTOFMEMORY); return FALSE; } plv->rcView.left = RECOMPUTE; // recompute view rect if (ListView_IsAutoArrangeView(plv)) { // Call off to the arrange function. ListView_OnArrange(plv, LVA_DEFAULT); if (!fInvalidateAll) { // Try to be smart and invalidate only what we need to. // Add a little logic to erase any message like no items found when // the view was previously empty... if (cTotalItemsOld < iItems) iItem = cTotalItemsOld; else iItem = iItems - 1; // Get the index if ((iItem >= 0) && (cTotalItemsOld > 0)) ListView_IInvalidateBelow(plv, iItem); else fInvalidateAll = TRUE; } } else { ListView_Recompute(plv); // if we have empty text and old count was zero... then we should redraw all if (plv->pszEmptyText && (cTotalItemsOld == 0) && (iItems > 0)) fInvalidateAll = TRUE; // Try to do smart invalidates... if (!fInvalidateAll) { // Try to be smart and invalidate only what we need to. if (cTotalItemsOld < iItems) iItem = cTotalItemsOld; else iItem = iItems - 1; // Get the index if (iItem >= 0) ListView_LRInvalidateBelow(plv, iItem, FALSE); } // We may try to resize the column ListView_MaybeResizeListColumns(plv, 0, ListView_Count(plv)-1); // For compatability we assume 0 for flags implies old type // of functionality and scrolls the important item into view. // If second bit is set, we leave the scroll position alone. if ((dwFlags & LVSICF_NOSCROLL) == 0) { // what is the important item iItem = (plv->iFocus >= 0) ? plv->iFocus : ListView_OnGetNextItem(plv, -1, LVNI_SELECTED); iItem = max(0, iItem); // make important item visable ListView_OnEnsureVisible(plv, iItem, FALSE); } } if (fInvalidateAll) InvalidateRect(plv->ci.hwnd, NULL, TRUE); ListView_UpdateScrollBars(plv); ListView_NotifyRecreate(plv); ListView_NotifyFocusEvent(plv); } else { frt = FALSE; } } else { if (plv->hdpaSubItems) { int iCol; for (iCol = plv->cCol - 1; iCol >= 0; iCol--) { HDPA hdpa = ListView_GetSubItemDPA(plv, iCol); if (hdpa) // this is optional, call backs don't have them DPA_Grow(hdpa, iItems); } } DPA_Grow(plv->hdpa, iItems); DPA_Grow(plv->hdpaZOrder, iItems); } return frt; } VOID ListView_InvalidateTTLastHit(LV* plv, int iNewHit) { if (plv->iTTLastHit == iNewHit) { plv->iTTLastHit = -1; if (plv->pszTip && plv->pszTip != LPSTR_TEXTCALLBACK) { plv->pszTip[0] = 0; } } } typedef struct { LV *plv; BOOL fSortIndices; PFNLVCOMPARE pfnCompare; LPARAM lParam; BOOL bPassLP; } LVSortInfo; int CALLBACK ListView_SortCallback(void * dw1, void * dw2, LPARAM lParam) { LISTITEM *pitem1; LISTITEM *pitem2; LVSortInfo *pSortInfo = (LVSortInfo *)lParam; ASSERT(!ListView_IsOwnerData(pSortInfo->plv)); // determine whether dw1 and dw2 are indices or the real items // and assign pitem? accordingly if (pSortInfo->fSortIndices) { pitem1 = ListView_GetItemPtr(pSortInfo->plv, PtrToUlong(dw1)); pitem2 = ListView_GetItemPtr(pSortInfo->plv, PtrToUlong(dw2)); } else { pitem1 = (LISTITEM *)dw1; pitem2 = (LISTITEM *)dw2; } if (!pSortInfo->pfnCompare) { // Treat NULL pszText like null string. LPCTSTR pszText1 = pitem1->pszText ? pitem1->pszText : c_szNULL; LPCTSTR pszText2 = pitem2->pszText ? pitem2->pszText : c_szNULL; // REARCHITECT: should allow callbacks in text if (pszText1 != LPSTR_TEXTCALLBACK && pszText2 != LPSTR_TEXTCALLBACK) { return lstrcmpi(pitem1->pszText, pitem2->pszText); } RIPMSG(0, "LVM_SORTITEM(EX): Cannot combine NULL callback with LPSTR_TEXTCALLBACK"); return -1; } else { if (pSortInfo->bPassLP) return pSortInfo->pfnCompare(pitem1->lParam, pitem2->lParam, pSortInfo->lParam); else { if (pSortInfo->fSortIndices) return pSortInfo->pfnCompare((LPARAM)dw1, (LPARAM)dw2, pSortInfo->lParam); else { // we want to sort by the indices, but all we've got are pointers to the items // and there is no way to get back from that pointer to an index RIPMSG(0, "LVM_SORTITEM(EX): Want to sort by indicies, but only have pointers"); return -1; } } } RIPMSG(0, "LVM_SORTITEM(EX): Didn't seem to sort by anything"); return -1; } LISTGROUP* ListView_FindGroupFromID(LV* plv, int iGroupId, int* piIndex) { if (plv->hdpaGroups) { int cGroups = DPA_GetPtrCount(plv->hdpaGroups); int iGroup; for (iGroup = 0; iGroup < cGroups; iGroup++) { LISTGROUP* pgrp = DPA_FastGetPtr(plv->hdpaGroups, iGroup); if (pgrp->iGroupId == iGroupId) { if (piIndex) *piIndex = iGroup; return pgrp; } } } return NULL; } BOOL ListView_VerifyGroupIdIsUnique(LV* plv, int iGroupId) { if (plv->hdpaGroups) { int cGroups = DPA_GetPtrCount(plv->hdpaGroups); int iGroup; for (iGroup = 0; iGroup < cGroups; iGroup++) { LISTGROUP* pgrp = DPA_FastGetPtr(plv->hdpaGroups, iGroup); if (pgrp->iGroupId == iGroupId) return FALSE; } } return TRUE; } int ListView_GroupIndexFromItem(LV* plv, LISTITEM* pitem) { if (LISTITEM_HASGROUP(pitem)) { int cGroups = DPA_GetPtrCount(plv->hdpaGroups); int iGroup; for (iGroup = 0; iGroup < cGroups; iGroup++) { LISTGROUP* pgrp = DPA_FastGetPtr(plv->hdpaGroups, iGroup); if (pgrp == pitem->pGroup) { return iGroup; } } } return -1; } BOOL ListView_RemoveItemFromItsGroup(LV* plv, LISTITEM* pitem) { if (LISTITEM_HASGROUP(pitem)) { LISTGROUP* pgrp = pitem->pGroup; int cItems = DPA_GetPtrCount(pgrp->hdpa); int iItem; for (iItem = 0; iItem < cItems; iItem++) { LISTITEM* pgitem = DPA_FastGetPtr(pgrp->hdpa, iItem); if (pgitem == pitem) { DPA_DeletePtr(pgrp->hdpa, iItem); return TRUE; } } } return FALSE; } BOOL ListView_FixupGroupsAfterSorting(LV *plv) { BOOL fRet = FALSE; int cGroups = DPA_GetPtrCount(plv->hdpaGroups); int *rgiGroupIds = LocalAlloc(LPTR, sizeof(int) * cGroups); // rgi will be where we keep the index in each group as we add items to them int *rgi = LocalAlloc(LPTR, sizeof(int) * cGroups); if (rgiGroupIds && rgi) { int i; int iMax = DPA_GetPtrCount(plv->hdpa); // Save away the group IDs, and temporary replace them with straight indices for (i=0; i < cGroups; i++) { LISTGROUP* pgrp = DPA_FastGetPtr(plv->hdpaGroups, i); rgiGroupIds[i] = pgrp->iGroupId; pgrp->iGroupId = i; } // Now all the items are sorted, and all we need to do it put them back in their // respective groups is sorted order for (i=0; i < iMax;i++) { LISTITEM *pitem = ListView_FastGetItemPtr(plv, i); LISTGROUP* pgrp = LISTITEM_GROUP(pitem); if (pgrp) { ASSERT(pgrp->hdpa); DPA_SetPtr(pgrp->hdpa, rgi[pgrp->iGroupId]++, pitem); } } #if DEBUG // At this point, we should still have the proper number of items in each group! for (i=0; i < cGroups; i++) { LISTGROUP* pgrp = DPA_FastGetPtr(plv->hdpaGroups, i); ASSERT(DPA_GetPtrCount(pgrp->hdpa) == rgi[i]); } #endif // Restore the proper GroupIds now for (i=0; i < cGroups; i++) { LISTGROUP* pgrp = DPA_FastGetPtr(plv->hdpaGroups, i); pgrp->iGroupId = rgiGroupIds[i]; } fRet = TRUE; } LocalFree(rgiGroupIds); LocalFree(rgi); return fRet; } BOOL ListView_SortAllColumns(LV* plv, LVSortInfo * psi) { BOOL fReturn; ASSERT(!ListView_IsOwnerData(plv)); ListView_InvalidateTTLastHit(plv, plv->iTTLastHit); // don't do this optimization if we will need the indices to sort by if (psi->bPassLP && ((!plv->hdpaSubItems) || !DPA_GetPtrCount(plv->hdpaSubItems))) { psi->fSortIndices = FALSE; fReturn = DPA_Sort(plv->hdpa, ListView_SortCallback, (LPARAM)psi); } else { // if we need to sort several hdpa's, create one DPA of just indices // and sort that, then fix up all the dpa's // initialize the hdpa with indices HDPA hdpa = DPA_Clone(plv->hdpa, NULL); fReturn = FALSE; if (hdpa) { int i, iMax; void **ph; void **pNewIndices; ASSERT(DPA_GetPtrCount(plv->hdpa) == DPA_GetPtrCount(hdpa)); ph = pNewIndices = DPA_GetPtrPtr(hdpa); iMax = DPA_GetPtrCount(hdpa); for (i = 0; i < iMax; ph++, i++) { *ph = IntToPtr(i); } psi->fSortIndices = TRUE; if (DPA_Sort(hdpa, ListView_SortCallback, (LPARAM)psi)) { ph = LocalAlloc(LPTR, sizeof(void *) * iMax); if (ph) { int j; void **pSubItems; // we could get here because bPassLP is false, even if we don't have subitems if (plv->hdpaSubItems && DPA_GetPtrCount(plv->hdpaSubItems)) { for (i = DPA_GetPtrCount(plv->hdpaSubItems) - 1; i >= 0; i--) { HDPA hdpaSubItem = ListView_GetSubItemDPA(plv, i); if (hdpaSubItem) { // make sure it's of the right size while (DPA_GetPtrCount(hdpaSubItem) < iMax) { if (DPA_InsertPtr(hdpaSubItem, iMax, NULL) == -1) goto Bail; } // actually copy across the dpa with the new indices pSubItems = DPA_GetPtrPtr(hdpaSubItem); for (j = 0; j < iMax; j++) { ph[j] = pSubItems[PtrToUlong(pNewIndices[j])]; } // finally, copy it all back to the pSubItems; memcpy(pSubItems, ph, sizeof(void *) * iMax); } } } // now do the main hdpa pSubItems = DPA_GetPtrPtr(plv->hdpa); for (j = 0; j < iMax; j++) { ph[j] = pSubItems[PtrToUlong(pNewIndices[j])]; } // finally, copy it all back to the pSubItems; memcpy(pSubItems, ph, sizeof(void *) * iMax); fReturn = TRUE; Bail: LocalFree(ph); } } DPA_Destroy(hdpa); } } if (fReturn && plv->fGroupView && plv->hdpaGroups && DPA_GetPtrCount(plv->hdpaGroups) > 0 && ListView_IsGroupedView(plv)) { fReturn = ListView_FixupGroupsAfterSorting(plv); } return fReturn; } DWORD ListView_OnApproximateViewRect(LV* plv, int iCount, int iWidth, int iHeight) { if (iCount == -1) iCount = ListView_Count(plv); if (iWidth == -1) iWidth = plv->sizeClient.cx; if (iHeight == -1) iHeight = plv->sizeClient.cy; return _ListView_ApproximateViewRect(plv, iCount, iWidth, iHeight); } DWORD ListView_OnSetLVRangeObject(LV* plv, int iWhich, ILVRange *plvrange) { ILVRange **pplvrange; switch (iWhich) { case LVSR_SELECTION: pplvrange = &plv->plvrangeSel; break; case LVSR_CUT: pplvrange = &plv->plvrangeCut; break; default: return FALSE; } if (*pplvrange) { // Release the old one (*pplvrange)->lpVtbl->Release(*pplvrange); } *pplvrange = plvrange; // Hold onto the pointer... if (plvrange) plvrange->lpVtbl->AddRef(plvrange); return TRUE; } BOOL ListView_OnSortItems(LV *plv, LPARAM lParam, PFNLVCOMPARE pfnCompare, BOOL bPassLP) { LVSortInfo SortInfo; LISTITEM *pitemFocused; SortInfo.pfnCompare = pfnCompare; SortInfo.lParam = lParam; SortInfo.plv = plv; SortInfo.bPassLP = bPassLP; if (ListView_IsOwnerData(plv)) { RIPMSG(0, "LVM_SORTITEMS: Invalid for owner-data listview"); return FALSE; } ListView_DismissEdit(plv, TRUE); // cancel edits // we're going to mess with the indices, so stash away the pointer to the // focused item. if (plv->iFocus != -1) { pitemFocused = ListView_GetItemPtr(plv, plv->iFocus); } else pitemFocused = NULL; if (ListView_SortAllColumns(plv, &SortInfo)) { // restore the focused item. if (pitemFocused) { int i; for (i = ListView_Count(plv) - 1; i >= 0 ; i--) { if (ListView_GetItemPtr(plv, i) == pitemFocused) { plv->iFocus = i; plv->iMark = i; } } } if (ListView_IsAutoArrangeView(plv)) { ListView_CommonArrange(plv, LVA_DEFAULT, plv->hdpa); } else { InvalidateRect(plv->ci.hwnd, NULL, TRUE); } // The items in the view have moved around; let apps know NotifyWinEvent(EVENT_OBJECT_REORDER, plv->ci.hwnd, OBJID_CLIENT, 0); return TRUE; } return FALSE; } void ListView_EnableWindow(LV* plv, BOOL wParam) { if (wParam) { if (plv->ci.style & WS_DISABLED) { plv->ci.style &= ~WS_DISABLED; // enabled ListView_OnSetBkColor(plv, plv->clrBkSave); } } else { if (!(plv->ci.style & WS_DISABLED)) { plv->clrBkSave = plv->clrBk; plv->ci.style |= WS_DISABLED; // disabled ListView_OnSetBkColor(plv, g_clrBtnFace); } } RedrawWindow(plv->ci.hwnd, NULL, NULL, RDW_INVALIDATE | RDW_ERASE); } BOOL ListView_IsItemVisibleI(LV* plv, int i) // Assumes parmss ok etc for speed. Called inside region calc code. { RECT rcBounds; // get bounding rect of item ListView_GetRects(plv, i, QUERY_DEFAULT, NULL, NULL, &rcBounds, NULL); // Should perf this up for multimonitor case where there are dead zones in work area... return RECTS_IN_SIZE(plv->sizeClient, rcBounds); } // Helper for ListView_RecalcRegion #define BitOn(lpbits, x, y, cx) (*((BYTE *)(lpbits + ((y * cx) + (x / 8)))) & (0x80 >> (x % 8))) void ListView_RecalcRegion(LV* plv, BOOL fForce, BOOL fRedraw) { HRGN hrgnUnion = NULL; HRGN hrgn = NULL; int i; HDC hdc = NULL; BYTE * lpBits = NULL; HBITMAP hbmp = NULL, hbmpOld = NULL; RECT rc, rcIcon = {0}; LISTITEM * pitem; BITMAP bm; // Bail out if we don't need to do any work if (!(plv->exStyle & LVS_EX_REGIONAL) || !ListView_RedrawEnabled(plv) || (plv->flags & LVF_INRECALCREGION)) return; // To prevent recursion plv->flags |= LVF_INRECALCREGION; if ((ListView_Count(plv) > 0)) { int cxIcon, cyIcon; int dxOffset, dyOffset; // Run through first to see if anything changed - bail if not! if (!fForce) { for (i = 0; i < ListView_Count(plv); i++) { pitem = ListView_FastGetItemPtr(plv, i); if (!ListView_IsItemVisibleI(plv, i)) { if (pitem->hrgnIcon == (HANDLE)-1 || !pitem->hrgnIcon) // Item was invisible and still is. Nothing changed. continue; if (pitem->hrgnIcon) { // Item was visible and now is invisible... Something // changed. pitem->ptRgn.x = RECOMPUTE; pitem->ptRgn.y = RECOMPUTE; DeleteObject(pitem->hrgnIcon); pitem->hrgnIcon = NULL; } } ListView_GetRects(plv, i, QUERY_DEFAULT, NULL, &rc, NULL, NULL); // If the location of the icon or the text rectangle have // changed, then we need to continue so that we can recalculate // the region. if ((pitem->pt.x != pitem->ptRgn.x) || (pitem->pt.y != pitem->ptRgn.y) || (!pitem->hrgnIcon) || !EqualRect((CONST RECT *)&pitem->rcTextRgn, (CONST RECT *)&rc)) goto changed; } // If we go through all the items and nothing changed, then // we can return without doing any work! ASSERT(i == ListView_Count(plv)); goto exit; changed:; } // Figure out the dimensions of the Icon rectangle - assumes // each Icon rectangle is the same size. ListView_GetRects(plv, 0, QUERY_DEFAULT, &rcIcon, NULL, NULL, NULL); // Center the icon in the rectangle CCGetIconSize(&plv->ci, plv->himl, &cxIcon, &cyIcon); dxOffset = (rcIcon.right - rcIcon.left - cxIcon) / 2; dyOffset = (rcIcon.bottom - rcIcon.top - cyIcon) / 2; cxIcon = rcIcon.right - rcIcon.left; cyIcon = rcIcon.bottom - rcIcon.top; if (!(hdc = CreateCompatibleDC(NULL)) || (!(hbmp = CreateBitmap(cxIcon, cyIcon, 1, 1, NULL)))) { goto BailOut; } GetObject(hbmp, sizeof(bm), &bm); if (!(lpBits = (BYTE *)GlobalAlloc(GPTR, bm.bmWidthBytes * bm.bmHeight))) goto BailOut; hbmpOld = SelectObject(hdc, hbmp); PatBlt(hdc, 0, 0, cxIcon, cyIcon, WHITENESS); if (hrgnUnion = CreateRectRgn(0, 0, 0, 0)) { for (i = 0; i < ListView_Count(plv); i++) { int x, y, iResult; BOOL fStarted = FALSE; LPRECT lprc; pitem = ListView_FastGetItemPtr(plv, i); if (pitem->pt.y == RECOMPUTE) continue; if (!ListView_IsItemVisibleI(plv, i)) { // ignore invisible items if (pitem->hrgnIcon && pitem->hrgnIcon!=(HANDLE)-1) { pitem->ptRgn.x = RECOMPUTE; pitem->ptRgn.y = RECOMPUTE; DeleteObject(pitem->hrgnIcon); pitem->hrgnIcon = (HANDLE)-1; } continue; } // Add the region for the icon text first ListView_GetRects(plv, i, QUERY_DEFAULT, &rcIcon, &rc, NULL, NULL); // If we're in edit mode always use rcTextRgn if (i == plv->iEdit) lprc = &pitem->rcTextRgn; else lprc = &rc; if (!(hrgn = CreateRectRgnIndirect(lprc))) goto Error; iResult = CombineRgn(hrgnUnion, hrgn, hrgnUnion, RGN_OR); DeleteObject(hrgn); if (iResult == ERROR) { // Error case - out of memory. Just select in a NULL region. Error: DeleteObject(hrgnUnion); hrgnUnion = NULL; break; } // Succeeded, copy the rectangle to rcTextRgn so we // can test against it in the future. Don't copy over // it if we are in edit mode, the rectangle is used to // store the edit window in that case. if (plv->iEdit != i) CopyRect(&pitem->rcTextRgn, (CONST RECT *)&rc); // Now create a region for the icon mask - or use the cached one if (!pitem->hrgnIcon || pitem->hrgnIcon == (HANDLE)-1) { // (pitem->pt.x != pitem->ptRgn.x) || // (pitem->pt.y != pitem->ptRgn.y)) HRGN hrgnIcon = NULL; // On slow machines, we'll just wrap the icon with a rectangle. But on // faster machines, we'll build a region that corresponds to the // mask for the icon so it looks sweet. if (g_fSlowMachine) { // Modify the rectangle slightly so it looks better // Glue the icon and text rectangles together rcIcon.bottom = rc.top; // Shrink the width of the rectangle so it's only as big as the icon itself InflateRect(&rcIcon, -dxOffset, 0); hrgnIcon = CreateRectRgnIndirect(&rcIcon); } else { // If the image isn't around, get it now. if (pitem->iImage == I_IMAGECALLBACK) { LV_ITEM item; item.iItem = i; item.iSubItem = 0; item.mask = LVIF_IMAGE; item.stateMask = LVIS_ALL; item.pszText = NULL; item.cchTextMax = 0; // BOGUS - do we need to worry about our state // getting messed up during the callback? ListView_OnGetItem(plv, &item); } ImageList_Draw(plv->himl, pitem->iImage, hdc, 0, 0, ILD_MASK | (pitem->state & LVIS_OVERLAYMASK)); GetBitmapBits(hbmp, bm.bmWidthBytes * bm.bmHeight, (void *)lpBits); for (y = 0; y < cyIcon; y++) { for (x = 0; x < cxIcon; x++) { if (!fStarted && !BitOn(lpBits, x, y, bm.bmWidthBytes)) { rc.left = x; rc.top = y; rc.bottom = y + 1; fStarted = TRUE; if (x == (cxIcon - 1)) { x++; goto AddIt; } else { continue; } } if (fStarted && BitOn(lpBits, x, y, bm.bmWidthBytes)) { AddIt: rc.right = x; // // Mirror the region so that the icons get displayed ok. [samera] // if (plv->ci.dwExStyle & RTL_MIRRORED_WINDOW) { int iLeft = rc.left; rc.left = (cxIcon - (rc.right+1)); rc.right = (cxIcon - (iLeft+1)); OffsetRect(&rc, rcIcon.left - dxOffset, rcIcon.top + dyOffset); } else OffsetRect(&rc, rcIcon.left + dxOffset, rcIcon.top + dyOffset); if (hrgn = CreateRectRgnIndirect(&rc)) { if (hrgnIcon || (hrgnIcon = CreateRectRgn(0, 0, 0, 0))) iResult = CombineRgn(hrgnIcon, hrgn, hrgnIcon, RGN_OR); else iResult = ERROR; DeleteObject(hrgn); } if (!hrgn || (iResult == ERROR)) { if (hrgnIcon) DeleteObject(hrgnIcon); goto Error; } fStarted = FALSE; } } } } if (hrgnIcon) { // Cache it since it takes a long time to build it if (pitem->hrgnIcon && pitem->hrgnIcon != (HANDLE)-1) DeleteObject(pitem->hrgnIcon); pitem->hrgnIcon = hrgnIcon; pitem->ptRgn = pitem->pt; // Add it to the accumulated window region if (ERROR == CombineRgn(hrgnUnion, hrgnIcon, hrgnUnion, RGN_OR)) goto Error; } } else { OffsetRgn(pitem->hrgnIcon, pitem->pt.x - pitem->ptRgn.x, pitem->pt.y - pitem->ptRgn.y); pitem->ptRgn = pitem->pt; if (ERROR == CombineRgn(hrgnUnion, pitem->hrgnIcon, hrgnUnion, RGN_OR)) goto Error; } } } } BailOut: if (lpBits) GlobalFree((HGLOBAL)lpBits); if (hbmp) { SelectObject(hdc, hbmpOld); DeleteObject(hbmp); } if (hdc) DeleteDC(hdc); // Windows takes ownership of the region when we select it in to the window SetWindowRgn(plv->ci.hwnd, hrgnUnion, fRedraw); exit: plv->flags &= ~LVF_INRECALCREGION; } HIMAGELIST CreateCheckBoxImagelist(HIMAGELIST himl, BOOL fTree, BOOL fUseColorKey, BOOL fMirror) { int cxImage, cyImage; HBITMAP hbm; HBITMAP hbmTemp; COLORREF clrMask = CLR_DEFAULT; HDC hdcDesk = GetDC(NULL); HDC hdc; RECT rc; int nImages = fTree ? 3 : 2; HTHEME hTheme; DTBGOPTS dtbg = {sizeof(DTBGOPTS), DTBG_DRAWSOLID, 0,}; // tell drawthemebackground to preserve the alpha channel if (!hdcDesk) return NULL; hdc = CreateCompatibleDC(hdcDesk); ReleaseDC(NULL, hdcDesk); if (!hdc) return NULL; hTheme = OpenThemeData(NULL, L"Button"); // Must protect against ImageList_GetIconSize failing in case app // gave us a bad himl if (himl && ImageList_GetIconSize(himl, &cxImage, &cyImage)) { // cxImage and cyImage are okay } else { cxImage = g_cxSmIcon; cyImage = g_cySmIcon; } himl = ImageList_Create(cxImage, cyImage, ILC_MASK | ILC_COLOR32, 0, nImages); hbm = CreateColorBitmap(cxImage * nImages, cyImage); // fill hbmTemp = SelectObject(hdc, hbm); rc.left = rc.top = 0; rc.bottom = cyImage; rc.right = cxImage * nImages; if (!hTheme) { if (fUseColorKey) { clrMask = RGB(255,000,255); // magenta if (clrMask == g_clrWindow) clrMask = RGB(000,000,255); // blue } else { clrMask = g_clrWindow; } // Don't fill the image with the mask when themes are on. We want this to // "Alpha blend to zero" or be clear. No transparent blt needed. FillRectClr(hdc, &rc, clrMask); } rc.right = cxImage; // now draw the real controls on InflateRect(&rc, -g_cxEdge, -g_cyEdge); rc.right++; rc.bottom++; if (fTree) OffsetRect(&rc, cxImage, 0); if (hTheme) { DrawThemeBackgroundEx(hTheme, hdc, BP_CHECKBOX, CBS_UNCHECKEDNORMAL, &rc, &dtbg); } else { DrawFrameControl(hdc, &rc, DFC_BUTTON, DFCS_BUTTONCHECK | DFCS_FLAT | (fUseColorKey? 0 : DFCS_TRANSPARENT)); } OffsetRect(&rc, cxImage, 0); // [msadek]; For the mirrored case, there is an off-by-one somewhere in MirrorIcon() or System API. // Since I will not be touching MirrorIcon() by any mean and no chance to fix a system API, // let's compensate for it here. if (fMirror) { OffsetRect(&rc, -1, 0); } if (hTheme) { DrawThemeBackgroundEx(hTheme, hdc, BP_CHECKBOX, CBS_CHECKEDNORMAL, &rc, &dtbg); } else { DrawFrameControl(hdc, &rc, DFC_BUTTON, DFCS_BUTTONCHECK | DFCS_FLAT | DFCS_CHECKED | (fUseColorKey? 0 : DFCS_TRANSPARENT)); } SelectObject(hdc, hbmTemp); if (fUseColorKey) { ImageList_AddMasked(himl, hbm, clrMask); } else { ImageList_Add(himl, hbm, NULL); } if (fMirror) { HICON hIcon = ImageList_ExtractIcon(0, himl, nImages-1); MirrorIcon(&hIcon, NULL); ImageList_ReplaceIcon(himl, nImages-1, hIcon); } DeleteDC(hdc); DeleteObject(hbm); if (hTheme) CloseThemeData(hTheme); return himl; } void ListView_InitCheckBoxes(LV* plv, BOOL fInitializeState) { HIMAGELIST himlCopy = (plv->himlSmall ? plv->himlSmall : plv->himl); HIMAGELIST himl; BOOL bMirror = FALSE; // [msadek], CheckBoxed need not to be mirrored. // mirroer it during imagelist creation time so that it displays correctly himl = CreateCheckBoxImagelist(himlCopy, FALSE, TRUE, IS_WINDOW_RTL_MIRRORED(plv->ci.hwnd)); ImageList_SetBkColor(himl, IsUsingCleartype()? (plv->clrBk) : (CLR_NONE)); ListView_OnSetImageList(plv, himl, LVSIL_STATE); if (fInitializeState) ListView_OnSetItemState(plv, -1, INDEXTOSTATEIMAGEMASK(1), LVIS_STATEIMAGEMASK); } void ListView_PopBubble(LV *plv) { if (plv->hwndToolTips) SendMessage(plv->hwndToolTips, TTM_POP, 0, 0); } DWORD ListView_ExtendedStyleChange(LV* plv, DWORD dwNewStyle, DWORD dwExMask) { DWORD dwOldStyle = plv->exStyle; // this will change the listview report size and painting algorithm // because of the leading edge, so need to re-update scroll bars // and repaint everything if (ListView_IsReportView(plv)) { ListView_RUpdateScrollBars(plv); InvalidateRect(plv->ci.hwnd, NULL, TRUE); } // Change of styles may also changes tooltip policy, so pop it ListView_PopBubble(plv); if (dwExMask) dwNewStyle = (plv->exStyle & ~ dwExMask) | (dwNewStyle & dwExMask); // Currently, LVS_EX_REGIONAL, LVS_EX_MULTIWORKAREAS, LVS_EX_HIDELABELS, and // LVS_EX_SINGLEROW are only supported for large icon view if (!ListView_IsIconView(plv)) { dwNewStyle &= ~(LVS_EX_REGIONAL | LVS_EX_MULTIWORKAREAS | LVS_EX_HIDELABELS | LVS_EX_SINGLEROW); } // LVS_EX_REGIONAL and LVS_EX_SINGLEROW are not supported for ownerdata if (ListView_IsOwnerData(plv)) { dwNewStyle &= ~(LVS_EX_REGIONAL | LVS_EX_SINGLEROW); } plv->exStyle = dwNewStyle; // do any invalidation or whatever is needed here. if ((dwOldStyle ^ dwNewStyle) & LVS_EX_HIDELABELS) { plv->rcView.left = RECOMPUTE; InvalidateRect(plv->ci.hwnd, NULL, TRUE); } if ((dwOldStyle ^ dwNewStyle) & LVS_EX_GRIDLINES) { if (ListView_IsReportView(plv)) { InvalidateRect(plv->ci.hwnd, NULL, TRUE); } } if ((dwOldStyle ^ dwNewStyle) & (LVS_EX_UNDERLINEHOT | LVS_EX_UNDERLINECOLD | LVS_EX_ONECLICKACTIVATE | LVS_EX_TWOCLICKACTIVATE | LVS_EX_SUBITEMIMAGES | LVS_EX_SNAPTOGRID)) { plv->rcView.left = RECOMPUTE; InvalidateRect(plv->ci.hwnd, NULL, TRUE); } if ((dwOldStyle ^ dwNewStyle) & LVS_EX_CHECKBOXES) { if (dwNewStyle & LVS_EX_CHECKBOXES) { ListView_InitCheckBoxes(plv, TRUE); } else { // destroy the check boxes! HIMAGELIST himl = ListView_OnSetImageList(plv, NULL, LVSIL_STATE); if (himl) ImageList_Destroy(himl); } } if ((dwOldStyle ^ dwNewStyle) & LVS_EX_FLATSB) { if (dwNewStyle & LVS_EX_FLATSB) { InitializeFlatSB(plv->ci.hwnd); if (plv->hwndHdr) { SetWindowBits(plv->hwndHdr, GWL_STYLE, HDS_FLAT, HDS_FLAT); } } else { if (plv->hwndHdr) { SetWindowBits(plv->hwndHdr, GWL_STYLE, HDS_FLAT, 0); } UninitializeFlatSB(plv->ci.hwnd); } } if ((dwOldStyle ^ dwNewStyle) & LVS_EX_REGIONAL) { g_fSlowMachine = FALSE; if (dwNewStyle & LVS_EX_REGIONAL) { ListView_RecalcRegion(plv, TRUE, TRUE); } else { int i; LISTITEM * pitem; // Delete all the cached regions, then NULL out our selected region. for (i = 0; i < ListView_Count(plv); i++) { pitem = ListView_FastGetItemPtr(plv, i); if (pitem->hrgnIcon && pitem->hrgnIcon!=(HANDLE)-1) { DeleteObject(pitem->hrgnIcon); } pitem->hrgnIcon = NULL; } SetWindowRgn(plv->ci.hwnd, (HRGN)NULL, TRUE); } InvalidateRect(plv->ci.hwnd, NULL, TRUE); } if ((dwOldStyle ^ dwNewStyle) & LVS_EX_SINGLEROW) { _ListView_RecomputeEx(plv, NULL, 0, TRUE); } if (ListView_IsDoubleBuffer(plv)) { InvalidateRect(plv->ci.hwnd, NULL, TRUE); } return dwOldStyle; } // Bug#94368 raymondc v6.0: Doesn't detect WM_WINDOWPOSCHANGING as a way // of being shown. NT5 defview has to hack around it pretty grossly. // Fix for v6.0. void LV_OnShowWindow(LV* plv, BOOL fShow) { if (fShow) { if (!(plv->flags & LVF_VISIBLE)) { plv->flags |= LVF_VISIBLE; if (plv->fGroupView) _ListView_RecomputeEx(plv, NULL, 0, FALSE); ListView_UpdateScrollBars(plv); } } else plv->flags &= ~LVF_VISIBLE; } LRESULT ListView_OnHelp(LV* plv, LPHELPINFO lpHelpInfo) { // If we're seeing WM_HELP because of our child header control, then // munge the HELPINFO structure to use the ListView's control id. // win\core\user\combo.c has similiar code to handle the child edit // control of a combo box. if ((lpHelpInfo != NULL) && (plv->wView == LV_VIEW_DETAILS) && (lpHelpInfo->iCtrlId == LVID_HEADER)) { lpHelpInfo->hItemHandle = plv->ci.hwnd; lpHelpInfo->iCtrlId = GetWindowID(plv->ci.hwnd); // Shouldn't have to do this: USER would have filled in the appropriate // context id by walking up the parent hwnd chain. //lpHelpInfo->dwContextId = GetContextHelpId(hwnd); } return DefWindowProc(plv->ci.hwnd, WM_HELP, 0, (LPARAM)lpHelpInfo); } DWORD ListView_OnSetIconSpacing(LV* plv, LPARAM lParam) { DWORD dwOld = MAKELONG(plv->cxIconSpacing, plv->cyIconSpacing); int cxIconSpacing, cyIconSpacing; if (lParam == (LPARAM)-1) { // go back to using defaults plv->flags &= ~LVF_ICONSPACESET; cxIconSpacing = (plv->cxIcon + (g_cxIconSpacing - g_cxIcon)); cyIconSpacing = (plv->cyIcon + (g_cyIconSpacing - g_cyIcon)); } else { if (LOWORD(lParam)) { cxIconSpacing = LOWORD(lParam); if (ListView_IsDPIScaled(plv)) CCDPIScaleX(&cxIconSpacing); } else { cxIconSpacing = plv->cxIconSpacing; } if (HIWORD(lParam)) { cyIconSpacing = HIWORD(lParam); if (ListView_IsDPIScaled(plv)) CCDPIScaleY(&cyIconSpacing); } else { cyIconSpacing = plv->cyIconSpacing; } plv->flags |= LVF_ICONSPACESET; } if ((cxIconSpacing != plv->cxIconSpacing) || (cyIconSpacing != plv->cyIconSpacing)) { plv->cxIconSpacing = cxIconSpacing; plv->cyIconSpacing = cyIconSpacing; plv->rcView.left = RECOMPUTE; // Recomputing is necessary except when snap-to-grid is toggled. Snap to grid assumes icon spacing // is the grid, however this is the only style that makes this assumption. if(!(plv->exStyle & LVS_EX_SNAPTOGRID)) _ListView_RecomputeEx(plv, NULL, 0, TRUE); plv->iFreeSlot = -1; } return dwOld; } BOOL ListView_OnSetCursorMsg(LV* plv) { if (plv->exStyle & (LVS_EX_ONECLICKACTIVATE|LVS_EX_TWOCLICKACTIVATE)) { if (plv->iHot != -1) { if (((plv->exStyle & LVS_EX_ONECLICKACTIVATE && plv->fOneClickOK)) || ListView_OnGetItemState(plv, plv->iHot, LVIS_SELECTED)) { if (!plv->hCurHot) { plv->hCurHot = LoadCursor(NULL, IDC_HAND); } SetCursor(plv->hCurHot); return TRUE; } } } return FALSE; } void ListView_OnSetHotItem(LV* plv, int iItem) { if (iItem != plv->iHot) { if ((plv->exStyle & LVS_EX_ONECLICKACTIVATE) || (plv->exStyle & LVS_EX_TWOCLICKACTIVATE)) // We only change visuals for hot with Underline { BOOL fSelectOnly; UINT uInvalidateFlags = RDW_INVALIDATE; BOOL fBlended = FALSE; // Check to see if the item we are making not is in a blended state if (iItem != -1) { // Cut is blended so we need to erase... fBlended = ListView_OnGetItemState(plv, iItem, LVIS_CUT); if (!fBlended) fBlended = ListView_OnGetItemState(plv, iItem, LVIS_SELECTED); } // If we need to erase either one then we erase both. if (plv->iHot != -1 && ListView_IsValidItemNumber(plv, plv->iHot) && !fBlended) { // Cut is blended so we need to erase... fBlended = ListView_OnGetItemState(plv, plv->iHot, LVIS_CUT); if (!fBlended) fBlended = ListView_OnGetItemState(plv, plv->iHot, LVIS_SELECTED); } if (ImageList_GetFlags(plv->himl) & ILC_COLOR32) fBlended = TRUE; // Affects only apply if double buffering if (ListView_IsDoubleBuffer(plv) || plv->fListviewShadowText || fBlended) { uInvalidateFlags |= RDW_ERASE; } fSelectOnly = ListView_FullRowSelect(plv); ListView_InvalidateItemEx(plv, plv->iHot, fSelectOnly, uInvalidateFlags, LVIF_TEXT | LVIF_IMAGE); ListView_InvalidateItemEx(plv, iItem, fSelectOnly, uInvalidateFlags, LVIF_TEXT | LVIF_IMAGE); } plv->iHot = iItem; } } BOOL fShouldFirstClickActivate() { static BOOL fInited = FALSE; static BOOL fActivate = TRUE; if (!fInited) { long cb = 0; if (RegQueryValue(HKEY_CURRENT_USER, TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Explorer\\NoFirstClickActivate"), NULL, &cb) == ERROR_SUCCESS) fActivate = FALSE; fInited = TRUE; } return fActivate; } BOOL ChildOfDesktop(HWND hwnd) { return IsChild(GetShellWindow(), hwnd); } void ListView_OnMouseMove(LV* plv, int x, int y, UINT uFlags) { int iItem; LV_HITTESTINFO ht; NMLISTVIEW nm; ht.pt.x = x; ht.pt.y = y; iItem = ListView_OnSubItemHitTest(plv, &ht); if (ht.iSubItem != 0) { // if we're not in full row select, // hitting on a subitem is like hitting on nowhere // also, in win95, ownerdraw fixed effectively had full row select if (!ListView_FullRowSelect(plv) && !(plv->ci.style & LVS_OWNERDRAWFIXED)) { iItem = -1; ht.flags = LVHT_NOWHERE; } } if (ht.flags & LVHT_NOWHERE || ht.flags & LVHT_ONITEMSTATEICON) { iItem = -1; // this is possible in the list mode (sigh) } nm.iItem = iItem; nm.iSubItem = ht.iSubItem; nm.uChanged = 0; nm.ptAction.x = x; nm.ptAction.y = y; if (!CCSendNotify(&plv->ci, LVN_HOTTRACK, &nm.hdr)) { #ifdef DEBUG if ((nm.iItem != -1) && nm.iSubItem != 0) nm.iItem = -1; #endif ListView_OnSetHotItem(plv, nm.iItem); // Ensure our cursor is correct now since the WM_SETCURSOR // message was already generated for this mouse event. ListView_OnSetCursorMsg(plv); // this lets us know when we've left an item // and can then reselect/toggle it on hover events if (iItem != plv->iNoHover) { plv->iNoHover = -1; } } } BOOL EditBoxHasFocus() { HWND hwndFocus = GetFocus(); if (hwndFocus) { if (SendMessage(hwndFocus, WM_GETDLGCODE, 0, 0) & DLGC_HASSETSEL) return TRUE; } return FALSE; } void ListView_OnMouseHover(LV* plv, int x, int y, UINT uFlags) { int iItem; BOOL bSelected; LV_HITTESTINFO ht; BOOL fControl; BOOL fShift; BOOL fNotifyReturn = FALSE; if (GetCapture() || !ChildOfActiveWindow(plv->ci.hwnd) || EditBoxHasFocus()) return; // ignore hover while editing or any captured (d/d) operation if (CCSendNotify(&plv->ci, NM_HOVER, NULL)) { return; } // REVIEW: right button implies no shift or control stuff // Single selection style also implies no modifiers //if (RIGHTBUTTON(keyFlags) || (plv->ci.style & LVS_SINGLESEL)) if ((plv->ci.style & LVS_SINGLESEL)) { fControl = FALSE; fShift = FALSE; } else { fControl = GetAsyncKeyState(VK_CONTROL) < 0; fShift = GetAsyncKeyState(VK_SHIFT) < 0; } ht.pt.x = x; ht.pt.y = y; iItem = ListView_OnHitTest(plv, &ht); if (iItem == -1 || iItem == plv->iNoHover) return; //before we hover select we launch any pending item //this prevents clicking on one item and hover selecting other before //the timer goes off which result in wrong item being launched if (plv->exStyle & LVS_EX_ONECLICKACTIVATE && plv->fOneClickHappened && plv->fOneClickOK) { HWND hwnd = plv->ci.hwnd; KillTimer(plv->ci.hwnd, IDT_ONECLICKHAPPENED); plv->fOneClickHappened = FALSE; CCSendNotify(&plv->ci, LVN_ITEMACTIVATE, &(plv->nmOneClickHappened.hdr)); if (!IsWindow(hwnd)) return; } plv->iNoHover = iItem; bSelected = ListView_OnGetItemState(plv, iItem, LVIS_SELECTED); if (ht.flags & (LVHT_ONITEMLABEL | LVHT_ONITEMICON)) { UINT keyFlags = 0; if (fShift) keyFlags |= MK_SHIFT; if (fControl) keyFlags |= MK_CONTROL; if (!bSelected) { // if it wasn't selected, we're about to select it... play // a little ditty for us... CCPlaySound(c_szSelect); } ListView_ButtonSelect(plv, iItem, keyFlags, bSelected); if (fControl) { ListView_SetFocusSel(plv, iItem, !fShift, FALSE, !fShift); } if (!fShift) plv->iMark = iItem; ListView_OnSetCursorMsg(plv); SetFocus(plv->ci.hwnd); // activate this window } } BOOL EqualRects(LPRECT prcNew, LPRECT prcOld, int nRects) { int i; for (i = 0; i < nRects; i++) if (!EqualRect(&prcNew[i], &prcOld[i])) return FALSE; return TRUE; } BOOL ListView_FindWorkArea(LV * plv, POINT pt, short * piWorkArea) { int iWork; for (iWork = 0; iWork < plv->nWorkAreas; iWork++) { if (PtInRect(&plv->prcWorkAreas[iWork], pt)) { *piWorkArea = (short)iWork; return TRUE; } } // (dli) default case is the primary work area *piWorkArea = 0; return FALSE; } void ListView_BullyIconsOnWorkarea(LV * plv, HDPA hdpaLostItems) { int ihdpa; int iFree = -1; // the last free slot number LVFAKEDRAW lvfd; LV_ITEM item; // Caller should've filtered this case out ASSERT(DPA_GetPtrCount(hdpaLostItems) > 0); // Set up in case caller is customdraw ListView_BeginFakeCustomDraw(plv, &lvfd, &item); item.mask = LVIF_PARAM; item.iSubItem = 0; // Go through my hdpa list of lost icons and try to place them within bound for (ihdpa = 0; ihdpa < DPA_GetPtrCount(hdpaLostItems); ihdpa++) { POINT ptNew, pt; RECT rcBound = {0}; int cxBound, cyBound; int iWidth, iHeight; int iItem; LISTITEM * pitem; iItem = PtrToUlong(DPA_GetPtr(hdpaLostItems, ihdpa)); pitem = ListView_FastGetItemPtr(plv, iItem); pt = pitem->pt; iWidth = RECTWIDTH(plv->prcWorkAreas[pitem->iWorkArea]); iHeight = RECTHEIGHT(plv->prcWorkAreas[pitem->iWorkArea]); ListView_GetRects(plv, iItem, QUERY_DEFAULT, NULL, NULL, &rcBound, NULL); cxBound = RECTWIDTH(rcBound); cyBound = RECTHEIGHT(rcBound); pt.x -= plv->prcWorkAreas[pitem->iWorkArea].left; pt.y -= plv->prcWorkAreas[pitem->iWorkArea].top; if (pt.x < (-cxBound / 2)) { ptNew.x = 0; } else if (pt.x > (iWidth - (cxBound / 2))) { ptNew.x = iWidth - cxBound; } else ptNew.x = pt.x; if (pt.y < (-cyBound/2)) { ptNew.y = 0; } else if (pt.y > (iHeight - (cyBound / 2))) { ptNew.y = iHeight - cyBound; } else ptNew.y = pt.y; if ((ptNew.x != pt.x) || (ptNew.y != pt.y)) { BOOL fUpdate; RECT rcTest; ptNew.x += plv->prcWorkAreas[pitem->iWorkArea].left; ptNew.y += plv->prcWorkAreas[pitem->iWorkArea].top; // See if the potential rectangle intersects other items. rcTest.left = ptNew.x - plv->ptOrigin.x; rcTest.top = ptNew.y - plv->ptOrigin.y; rcTest.right = rcTest.left + cxBound; rcTest.bottom = rcTest.top + cyBound; item.iItem = iItem; item.lParam = pitem->lParam; ListView_BeginFakeItemDraw(&lvfd); if (!ListView_IsCleanRect(plv, &rcTest, iItem, QUERY_DEFAULT, &fUpdate, lvfd.nmcd.nmcd.hdc)) { // doh! We hit another item, let's try to find an available location // for this item BOOL fUpdateSB; BOOL fAppendAtEnd = FALSE; int iWidth, iHeight; int cSlots = ListView_GetSlotCountEx(plv, FALSE, pitem->iWorkArea, &iWidth, &iHeight); iFree = ListView_FindFreeSlot(plv, iItem, iFree + 1, cSlots, QUERY_DEFAULT, &fUpdateSB, &fAppendAtEnd, lvfd.nmcd.nmcd.hdc, iWidth, iHeight); if (iFree == -1) goto SetFirstGuess; ListView_SetIconPos(plv, pitem, iFree, cSlots); ListView_EndFakeItemDraw(&lvfd); continue; } SetFirstGuess: ListView_EndFakeItemDraw(&lvfd); ListView_OnSetItemPosition(plv, iItem, ptNew.x, ptNew.y); } } ListView_EndFakeCustomDraw(&lvfd); } #define DPA_LAST 0x7fffffff // // ListView_OnSetWorkAreas // // set the "work areas" for the list view. // the "work areas" are a group of sub rectanges of the list view client rect // where icons are aranged, and parked by default. // void ListView_OnSetWorkAreas(LV* plv, int nWorkAreas, LPRECT prc) { int iWork; HDPA hdpaLostItems = NULL; RECT rcOldWorkAreas[LV_MAX_WORKAREAS]; BOOL bAutoArrange = plv->ci.style & LVS_AUTOARRANGE; int nOldWorkAreas = plv->nWorkAreas; if (nOldWorkAreas > 0) { ASSERT(plv->prcWorkAreas != NULL); memcpy(&rcOldWorkAreas[0], &plv->prcWorkAreas[0], sizeof(RECT) * nOldWorkAreas); } // for the mirrored case, the coordinates are reversed. IsRectEmpty() will always succeed if (nWorkAreas == 0 || prc == NULL || ((IsRectEmpty(prc)) && !(plv->ci.dwExStyle & RTL_MIRRORED_WINDOW))) plv->nWorkAreas = 0; else { plv->nWorkAreas = min(nWorkAreas, LV_MAX_WORKAREAS); if (plv->prcWorkAreas == NULL) plv->prcWorkAreas = (LPRECT)LocalAlloc(LPTR, sizeof(RECT) * LV_MAX_WORKAREAS); if (plv->prcWorkAreas == NULL) return; //Should we check if they intersect? This problem is sort of // solved (or made more confusing) by ListView_GetFreeSlot since it checks all of the icons for // intersection instead of just the ones in the workarea. for (iWork = 0; iWork < plv->nWorkAreas; iWork++) CopyRect(&plv->prcWorkAreas[iWork], &prc[iWork]); } // We don't support workareas for owner-data because our icon placement // algorithm (ListView_IGetRectsOwnerData) completely ignores workareas // and just dumps the icons in a rectangular array starting at (0,0). if (!ListView_IsOwnerData(plv) && plv->nWorkAreas > 0 && ((plv->nWorkAreas != nOldWorkAreas) || (!EqualRects(&plv->prcWorkAreas[0], &rcOldWorkAreas[0], nOldWorkAreas)))) { int iItem; LISTITEM * pitem; // // Subtle - ListView_Recompute cleans up all the RECOMPUTE icons, // but in order to do that, it needs to have valid work area // rectangles. So the call must happen after the CopyRect but before // the loop that checks the icon positions. // ListView_Recompute(plv); for (iItem = 0; iItem < ListView_Count(plv); iItem++) { pitem = ListView_FastGetItemPtr(plv, iItem); if (pitem->pt.x == RECOMPUTE || pitem->pt.y == RECOMPUTE) { // ListView_Recompute should've fixed these if we were in // an iconical view. ASSERT(!(ListView_IsIconView(plv) || ListView_IsSmallView(plv))); continue; } // Try to move me to the same location relative to the same workarea. // This will give the cool shift effect when tools bars take the border areas. // And we only want to do this for the workareas that changed // Don't bully the icons on the workareas, Autoarrange will do the work for us if (nOldWorkAreas > 0) { int iOldWorkArea; iOldWorkArea = pitem->iWorkArea; if (iOldWorkArea >= plv->nWorkAreas) { // My workarea is gone, put me on the primary workarea i.e. #0 pitem->iWorkArea = 0; if (!bAutoArrange) { // If this item point location is already in the new primary workarea, // move it out, and let ListView_BullyIconsOnWorkarea arrange it to the // right place. NOTE: this could happen in the case the old secondary monitor // is to the left of the old primary monitor, and user kills the secondary monitor if (PtInRect(&plv->prcWorkAreas[0], pitem->pt)) { pitem->pt.x = plv->prcWorkAreas[0].right + 1; plv->iFreeSlot = -1; // an item moved -- old slot info is invalid } goto InsertLostItemsArray; } } else if ((!bAutoArrange) && (!EqualRect(&plv->prcWorkAreas[iOldWorkArea], &rcOldWorkAreas[iOldWorkArea]))) { RECT rcBound = {0}; POINT ptCenter; pitem->pt.x += plv->prcWorkAreas[iOldWorkArea].left - rcOldWorkAreas[iOldWorkArea].left; pitem->pt.y += plv->prcWorkAreas[iOldWorkArea].top - rcOldWorkAreas[iOldWorkArea].top; // Use the center of this icon to determine whether it's out of bound ListView_GetRects(plv, iItem, QUERY_DEFAULT, NULL, NULL, &rcBound, NULL); ptCenter.x = pitem->pt.x + RECTWIDTH(rcBound) / 2; ptCenter.y = pitem->pt.y + RECTHEIGHT(rcBound) / 2; // If this shifted me out of bounds, register to be bullied on the workarea if (!PtInRect(&plv->prcWorkAreas[iOldWorkArea], ptCenter)) { InsertLostItemsArray: if (!hdpaLostItems) { hdpaLostItems = DPA_Create(4); } if (hdpaLostItems) DPA_InsertPtr(hdpaLostItems, DPA_LAST, IntToPtr(iItem)); } } } else { // My first time in a multi-workarea system, so find out my workarea if (!ListView_FindWorkArea(plv, pitem->pt, &(pitem->iWorkArea)) && !bAutoArrange) goto InsertLostItemsArray; } if ((plv->exStyle & LVS_EX_REGIONAL) && (pitem->hrgnIcon)) { if (pitem->hrgnIcon != (HANDLE)-1) DeleteObject(pitem->hrgnIcon); pitem->hrgnIcon = NULL; } } if (hdpaLostItems) { ASSERT(!bAutoArrange); if (DPA_GetPtrCount(hdpaLostItems) > 0) ListView_BullyIconsOnWorkarea(plv, hdpaLostItems); DPA_Destroy(hdpaLostItems); } if (plv->exStyle & LVS_EX_REGIONAL) ListView_RecalcRegion(plv, TRUE, TRUE); if (ListView_IsSmallView(plv) || ListView_IsIconView(plv)) ListView_ArrangeOrSnapToGrid(plv); } RedrawWindow(plv->ci.hwnd, NULL, NULL, RDW_INVALIDATE | RDW_ERASE); } void ListView_OnGetNumberOfWorkAreas(LV* plv, int * pnWorkAreas) { if (pnWorkAreas) *pnWorkAreas = plv->nWorkAreas; } void ListView_OnGetWorkAreas(LV* plv, int nWorkAreas, LPRECT prc) { int i; if (prc == NULL) return; for (i = 0; i < min(plv->nWorkAreas, nWorkAreas); i++) { if (i < plv->nWorkAreas) CopyRect(&prc[i], &plv->prcWorkAreas[i]); else // Set the workareas to all zeros if we don't have it. ZeroMemory(&prc[i], sizeof(RECT)); } } // test an item to see if it is unfolded (because it is focused) BOOL ListView_IsItemUnfolded(LV *plv, int item) { return plv && (item >= 0) && ListView_IsIconView(plv) && (plv->flags & LVF_UNFOLDED) && (plv->iFocus == item); } BOOL ListView_IsItemUnfoldedPtr(LV *plv, LISTITEM *pitem) { return plv && pitem && ListView_IsIconView(plv) && (plv->flags & LVF_UNFOLDED) && (pitem->state & LVIS_FOCUSED); } // Returns TRUE if unfolding the item will be worthwhile BOOL ListView_GetUnfoldedRect(LV* plv, int iItem, RECT *prc) { ListView_GetRects(plv, iItem, QUERY_DEFAULT, NULL, prc, NULL, NULL); return ListView_UnfoldRects(plv, iItem, NULL, prc, NULL, NULL); } BOOL ListView_OnSetGroupInfoInternal(LV* plv, PLVGROUP plvgrp, LISTGROUP* pgrp) { if (plvgrp == NULL || plvgrp->cbSize < sizeof(LVGROUP)) { return FALSE; } if (plvgrp->mask & LVGF_STATE) { if ((plvgrp->state & LVGS_MASK) != plvgrp->state) return FALSE; pgrp->state = plvgrp->state; } if (plvgrp->mask & LVGF_ALIGN) { if ((plvgrp->uAlign & LVGA_ALIGN_MASK) != plvgrp->uAlign) return FALSE; pgrp->uAlign = plvgrp->uAlign; } else { pgrp->uAlign = LVGA_HEADER_LEFT; } if (ListView_VerifyGroupIdIsUnique(plv, plvgrp->iGroupId)) { pgrp->iGroupId = plvgrp->iGroupId; } else { return FALSE; } if (plvgrp->mask & LVGF_HEADER) { Str_SetPtr(&pgrp->pszHeader, plvgrp->pszHeader); } if (plvgrp->mask & LVGF_FOOTER) { Str_SetPtr(&pgrp->pszFooter, plvgrp->pszFooter); } // Update the group. InvalidateRect(plv->ci.hwnd, &pgrp->rc, TRUE); return TRUE; } int ListView_OnSetGroupInfo(LV* plv, int iGroupId, PLVGROUP plvgrp) { LISTGROUP* pgrp = ListView_FindGroupFromID(plv, iGroupId, NULL); if (pgrp) { ListView_OnSetGroupInfoInternal(plv, plvgrp, pgrp); return iGroupId; } return -1; } int ListView_OnGetGroupInfo(LV* plv, int iGroupId, PLVGROUP plvgrp) { LISTGROUP* pgrp = ListView_FindGroupFromID(plv, iGroupId, NULL); if (plvgrp != NULL && plvgrp->cbSize >= sizeof(LVGROUP) && pgrp) { if (plvgrp->mask & LVGF_HEADER) { plvgrp->pszHeader = pgrp->pszHeader; } if (plvgrp->mask & LVGF_FOOTER) { plvgrp->pszFooter = pgrp->pszFooter; } if (plvgrp->mask & LVGF_STATE) { plvgrp->state = pgrp->state & plvgrp->stateMask; } if (plvgrp->mask & LVGF_ALIGN) { plvgrp->uAlign = pgrp->uAlign; } if (plvgrp->mask & LVGF_GROUPID) { plvgrp->iGroupId = pgrp->iGroupId; } return iGroupId; } return -1; } LISTGROUP* ListView_CreateGroup(LV* plv, PLVGROUP plvgrp) { LISTGROUP* pgrp; // Validate Group if (plvgrp == NULL || plvgrp->cbSize < sizeof(LVGROUP)) { return NULL; } if (!(plvgrp->mask & LVGF_GROUPID)) { // Have to have a group id... return NULL; } pgrp = LocalAlloc(LPTR, sizeof(LISTGROUP)); if (pgrp) { if (!ListView_OnSetGroupInfoInternal(plv, plvgrp, pgrp)) { LocalFree(pgrp); return NULL; } pgrp->hdpa = DPA_Create(5); SetRect(&pgrp->rc, 0, 0, 0, 0); } return pgrp; } void ListView_FreeGroupItem(LISTGROUP* pgrp) { DPA_Destroy(pgrp->hdpa); Str_SetPtr(&pgrp->pszFooter, NULL); Str_SetPtr(&pgrp->pszHeader, NULL); LocalFree(pgrp); } LISTGROUP* ListView_FindFirstVisibleGroup(LV* plv) { LISTGROUP* pgrp = NULL; int iGroup; int cGroups = DPA_GetPtrCount(plv->hdpaGroups); // Find the first group with an item in it. for (iGroup = 0; iGroup < cGroups; iGroup++) { pgrp = DPA_FastGetPtr(plv->hdpaGroups, iGroup); if (DPA_GetPtrCount(pgrp->hdpa) > 0) break; } return pgrp; } LRESULT ListView_OnInsertGroup(LV* plv, int iGroup, PLVGROUP plvgrp) { int iInsert = -1; LISTGROUP* pgrp = ListView_CreateGroup(plv, plvgrp); if (!pgrp) { return -1; } if (iGroup == -1) { iGroup = DA_LAST; } if (plv->hdpaGroups == NULL) plv->hdpaGroups = DPA_Create(4); if (plv->hdpaGroups) iInsert = DPA_InsertPtr(plv->hdpaGroups, iGroup, pgrp); if (iInsert == -1) { ListView_FreeGroupItem(pgrp); } plv->rcView.left = RECOMPUTE; InvalidateRect(plv->ci.hwnd, NULL, TRUE); return iInsert; } LRESULT ListView_OnRemoveGroup(LV* plv, int iGroupId) { int iIndex; LISTGROUP* pgrp = ListView_FindGroupFromID(plv, iGroupId, &iIndex); if (pgrp) { int cItems = DPA_GetPtrCount(pgrp->hdpa); int iItem; for (iItem = 0; iItem < cItems; iItem++) { LISTITEM* pitem = DPA_FastGetPtr(pgrp->hdpa, iItem); if (pitem) { LISTITEM_SETHASNOTASKEDFORGROUP(pitem); } } ListView_FreeGroupItem(pgrp); DPA_DeletePtr(plv->hdpaGroups, iIndex); plv->rcView.left = RECOMPUTE; InvalidateRect(plv->ci.hwnd, NULL, TRUE); return iIndex; } return -1; } int CALLBACK DestroyGroups(void* pv, void* pvData) { LISTGROUP* pgrp = (LISTGROUP*)pv; ListView_FreeGroupItem(pgrp); return 1; } LRESULT ListView_OnRemoveAllGroups(LV* plv) { if (!ListView_IsOwnerData(plv) && plv->hdpaGroups) { int i; int cItems = ListView_Count(plv); plv->fGroupView = FALSE; for (i = 0; i < cItems; i++) { LISTITEM* pitem = DPA_FastGetPtr(plv->hdpa, i); if (pitem) LISTITEM_SETHASNOTASKEDFORGROUP(pitem); } DPA_DestroyCallback(plv->hdpaGroups, DestroyGroups, NULL); plv->hdpaGroups = NULL; plv->rcView.left = RECOMPUTE; InvalidateRect(plv->ci.hwnd, NULL, TRUE); } return 1; } LRESULT ListView_OnSetGroupMetrics(LV* plv, PLVGROUPMETRICS pgm) { BOOL fRecompute = FALSE; if (pgm->mask & LVGMF_BORDERSIZE) { plv->rcBorder.left = pgm->Left; plv->rcBorder.top = pgm->Top; plv->rcBorder.right = pgm->Right; plv->rcBorder.bottom = pgm->Bottom; fRecompute = TRUE; } if (pgm->mask & LVGMF_BORDERCOLOR) { plv->crTop = pgm->crTop; plv->crLeft = pgm->crLeft; plv->crRight = pgm->crRight; plv->crBottom = pgm->crBottom; } if (pgm->mask & LVGMF_TEXTCOLOR) { plv->crHeader = pgm->crHeader; plv->crFooter = pgm->crFooter; } if (fRecompute) { plv->rcView.left = RECOMPUTE; ListView_Recompute(plv); ListView_UpdateScrollBars(plv); } InvalidateRect(plv->ci.hwnd, NULL, TRUE); return 1; } LRESULT ListView_OnGetGroupMetrics(LV* plv, PLVGROUPMETRICS pgm) { if (pgm->mask & LVGMF_BORDERSIZE) { pgm->Left = plv->rcBorder.left; pgm->Top = plv->rcBorder.top; pgm->Right = plv->rcBorder.right; pgm->Bottom = plv->rcBorder.bottom; } if (pgm->mask & LVGMF_BORDERCOLOR) { pgm->crTop = plv->crTop; pgm->crLeft = plv->crLeft; pgm->crRight = plv->crRight; pgm->crBottom = plv->crBottom; } if (pgm->mask & LVGMF_TEXTCOLOR) { pgm->crHeader = plv->crHeader; pgm->crFooter= plv->crFooter; } return 1; } typedef struct { PFNLVGROUPCOMPARE pfnCompare; void * pvData; } SORTGROUPDATA; int CALLBACK pfnGroupSort(LPARAM one, LPARAM two, LPARAM pvData) { SORTGROUPDATA* psg = (SORTGROUPDATA*)pvData; LISTGROUP* pgrp1 = (LISTGROUP*)one; LISTGROUP* pgrp2 = (LISTGROUP*)two; if (!one) return 1; if (!two) return -1; return psg->pfnCompare(pgrp1->iGroupId, pgrp2->iGroupId, psg->pvData); } LRESULT ListView_OnSortGroups(LV* plv, PFNLVGROUPCOMPARE pfnCompare, void * pvData) { if (plv->hdpaGroups) { SORTGROUPDATA sg; sg.pfnCompare = pfnCompare; sg.pvData = pvData; DPA_Sort(plv->hdpaGroups, (PFNDPACOMPARE)pfnGroupSort, (LPARAM)&sg); plv->rcView.left = RECOMPUTE; ListView_Recompute(plv); InvalidateRect(plv->ci.hwnd, NULL, TRUE); ListView_UpdateScrollBars(plv); return 1; } return 0; } LRESULT ListView_OnInsertGroupSorted(LV* plv, LVINSERTGROUPSORTED* pinsert) { int iInsertIndex = -1; SORTGROUPDATA sg; LISTGROUP* pgrp = ListView_CreateGroup(plv, &pinsert->lvGroup); sg.pfnCompare = pinsert->pfnGroupCompare; sg.pvData = pinsert->pvData; if (pgrp) { if (plv->hdpaGroups == NULL) plv->hdpaGroups = DPA_Create(4); if (plv->hdpaGroups) { iInsertIndex = DPA_SortedInsertPtr(plv->hdpaGroups, pgrp, 0, (PFNDPACOMPARE)pfnGroupSort, (LPARAM)&sg, DPAS_INSERTAFTER, pgrp); } if (iInsertIndex == -1) ListView_FreeGroupItem(pgrp); plv->rcView.left = RECOMPUTE; InvalidateRect(plv->ci.hwnd, NULL, TRUE); } return iInsertIndex; } BOOL ListView_OnSetTileViewInfo(LV* plv, PLVTILEVIEWINFO pTileViewInfo) { BOOL bRecompute = FALSE; if (!pTileViewInfo || (pTileViewInfo->cbSize != sizeof(LVTILEVIEWINFO))) return FALSE; if (pTileViewInfo->dwMask & LVTVIM_COLUMNS) { if (plv->cSubItems != pTileViewInfo->cLines) { bRecompute = TRUE; plv->cSubItems = pTileViewInfo->cLines; } } if (pTileViewInfo->dwMask & LVTVIM_TILESIZE) { DWORD dwTileFlags = pTileViewInfo->dwFlags & (LVTVIF_FIXEDHEIGHT | LVTVIF_FIXEDWIDTH); if (plv->dwTileFlags != dwTileFlags) { plv->dwTileFlags = dwTileFlags; bRecompute = TRUE; } if (ListView_IsDPIScaled(plv)) { CCDPIScaleX(&pTileViewInfo->sizeTile.cx); CCDPIScaleY(&pTileViewInfo->sizeTile.cy); } if ((plv->dwTileFlags & LVTVIF_FIXEDHEIGHT) && plv->sizeTile.cy != pTileViewInfo->sizeTile.cy) { plv->sizeTile.cy = pTileViewInfo->sizeTile.cy; bRecompute = TRUE; } if ((plv->dwTileFlags & LVTVIF_FIXEDWIDTH) && plv->sizeTile.cx != pTileViewInfo->sizeTile.cx) { plv->sizeTile.cx = pTileViewInfo->sizeTile.cx; bRecompute = TRUE; } } if ((pTileViewInfo->dwMask & LVTVIM_LABELMARGIN) && !EqualRect(&plv->rcTileLabelMargin, &pTileViewInfo->rcLabelMargin)) { plv->rcTileLabelMargin = pTileViewInfo->rcLabelMargin; bRecompute = TRUE; } if (bRecompute) { ListView_RecalcTileSize(plv); plv->rcView.left = RECOMPUTE; ListView_Recompute(plv); InvalidateRect(plv->ci.hwnd, NULL, TRUE); } return TRUE; } BOOL ListView_OnGetTileViewInfo(LV* plv, PLVTILEVIEWINFO pTileViewInfo) { if (!pTileViewInfo || (pTileViewInfo->cbSize != sizeof(LVTILEVIEWINFO))) return FALSE; if (pTileViewInfo->dwMask & LVTVIM_COLUMNS) { pTileViewInfo->cLines = plv->cSubItems; } if (pTileViewInfo->dwMask & LVTVIM_TILESIZE) { pTileViewInfo->dwFlags = plv->dwTileFlags; pTileViewInfo->sizeTile = plv->sizeTile; } if (pTileViewInfo->dwMask & LVTVIM_LABELMARGIN) { pTileViewInfo->rcLabelMargin = plv->rcTileLabelMargin; } return TRUE; } BOOL ListView_OnSetTileInfo(LV* plv, PLVTILEINFO pTileInfo) { LVITEM lvi; if (!pTileInfo || (pTileInfo->cbSize != sizeof(LVTILEINFO))) return FALSE; lvi.mask = LVIF_COLUMNS; lvi.cColumns = pTileInfo->cColumns; lvi.puColumns = pTileInfo->puColumns; lvi.iSubItem = 0; lvi.iItem = pTileInfo->iItem; return ListView_OnSetItem(plv, &lvi); } BOOL ListView_OnGetTileInfo(LV* plv, PLVTILEINFO pTileInfo) { LVITEM lvi; if (!pTileInfo || (pTileInfo->cbSize != sizeof(LVTILEINFO))) return FALSE; lvi.mask = LVIF_COLUMNS; lvi.iSubItem = 0; lvi.iItem = pTileInfo->iItem; lvi.cColumns = pTileInfo->cColumns; lvi.puColumns = pTileInfo->puColumns; if (ListView_OnGetItem(plv, &lvi)) { pTileInfo->cColumns = lvi.cColumns; return TRUE; } return FALSE; } LRESULT ListView_OnSetInsertMark(LV* plv, LPLVINSERTMARK plvim) { if (plvim->cbSize != sizeof(LVINSERTMARK)) return 0; if (plvim->iItem != plv->iInsertItem || BOOLIFY(plv->fInsertAfter) != BOOLIFY(plvim->dwFlags & LVIM_AFTER)) { if (plv->iInsertItem != -1) ListView_InvalidateMark(plv); plv->iInsertItem = plvim->iItem; plv->fInsertAfter = BOOLIFY(plvim->dwFlags & LVIM_AFTER); if (plv->iInsertItem != -1) ListView_InvalidateMark(plv); } return 1; } LRESULT ListView_OnSetInfoTip(LV *plv, PLVSETINFOTIP plvSetInfoTip) { LPWSTR pszText = NULL; LPWSTR pszProduced = NULL; LRESULT lRet = 0; // Check size and flags. MBZ for now. if (plvSetInfoTip->cbSize == sizeof(LVSETINFOTIP) && plvSetInfoTip->dwFlags == 0 && plvSetInfoTip->pszText != NULL) { pszText = plvSetInfoTip->pszText; // If we are still looking at the same item, then set its text, and pop up the tip. if (plvSetInfoTip->iItem == plv->iTTLastHit && plvSetInfoTip->iSubItem == plv->iTTLastSubHit) { TCHAR szBuf[INFOTIPSIZE]; BOOL bItemUnfolded; BOOL fInfoTip = FALSE; szBuf[0] = 0; // preload the default tip text for folded items. bItemUnfolded = ListView_IsItemUnfolded2(plv, plv->iTTLastHit, plv->iTTLastSubHit, szBuf, ARRAYSIZE(szBuf)); if (ListView_IsInfoTip(plv) && plv->iTTLastSubHit == 0) { if (*pszText && lstrcmp(szBuf, pszText) != 0) { // App changed something - there is a real infotip fInfoTip = TRUE; } } else { pszText = szBuf; } // // Set the margins now before the TTN_SHOW because it will be too late then. // // We want fat margins if we're an infotip, thin margins if we're an // in-place tooltip. // if (fInfoTip) { static const RECT rcMargin = {4, 4, 4, 4}; SendMessage(plv->hwndToolTips, TTM_SETMARGIN, 0, (LPARAM)&rcMargin); CCSetInfoTipWidth(plv->ci.hwnd, plv->hwndToolTips); plv->fPlaceTooltip = FALSE; // Set it to TRUE only if Unfolding tip is set } else { static const RECT rcMargin = {0, 0, 0, 0}; plv->fPlaceTooltip = TRUE; SendMessage(plv->hwndToolTips, TTM_SETMARGIN, 0, (LPARAM)&rcMargin); CCResetInfoTipWidth(plv->ci.hwnd, plv->hwndToolTips); } Str_Set(&plv->pszTip, pszText); // Re-display tooltip. If tracking, call tracking start code (same as timer code) if (!ListView_IsKbdTipTracking(plv)) lRet = SendMessage(plv->hwndToolTips, TTM_POPUP, 0, 0); else ListView_OnTimer(plv, IDT_TRACKINGTIP); } if (pszProduced) FreeProducedString(pszProduced); } return lRet; } LRESULT ListView_OnNotify(LV* plv, WPARAM wParam, LPNMHDR pnmh) { // we can't switch on the control ID because the tooltip is a WS_POPUP window // and does not have a control ID. (header and tooltip both have 0 as ID) if (plv->hwndHdr && (plv->hwndHdr == pnmh->hwndFrom)) { // this is a notify for the header, deal with it as needed return ListView_HeaderNotify(plv, (HD_NOTIFY *)pnmh); } else if (plv->hwndToolTips && (plv->hwndToolTips == pnmh->hwndFrom)) { // implement unfolding the text for items as well as info tip support switch (pnmh->code) { case TTN_NEEDTEXT: { POINT pt; UINT uFlags; int iNewHit; int iNewSubHit; NMTTDISPINFO *pttt = (NMTTDISPINFO *)pnmh; // If keyboard tracking, do not hit test based on last cursor position if (ListView_IsKbdTipTracking(plv)) { RECT rcItem; ListView_GetItemRect(plv->ci.hwnd, plv->iTracking, &rcItem, LVIR_LABEL); pt.x = rcItem.left; pt.y = rcItem.top; } else GetMessagePosClient(plv->ci.hwnd, &pt); iNewHit = _ListView_ItemHitTest(plv, pt.x, pt.y, &uFlags, &iNewSubHit); if (iNewHit != plv->iTTLastHit || iNewSubHit != plv->iTTLastSubHit) { plv->fPlaceTooltip = FALSE; // Set it to TRUE only if Unfolding tip is set Str_Set(&plv->pszTip, NULL); // clear the old tip plv->iTTLastHit = iNewHit; plv->iTTLastSubHit = iNewSubHit; if ((iNewHit >= 0) && (plv->iEdit == -1)) { TCHAR szBuf[INFOTIPSIZE], szBuf2[INFOTIPSIZE]; BOOL bItemUnfolded; BOOL fInfoTip = FALSE; LPTSTR pszTip = szBuf; // Use this one first szBuf[0] = 0; szBuf2[0] = 0; // preload the tip text for folded items. this // may be overridden by callback below bItemUnfolded = ListView_IsItemUnfolded2(plv, plv->iTTLastHit, plv->iTTLastSubHit, szBuf, ARRAYSIZE(szBuf)); lstrcpyn(szBuf2, szBuf, ARRAYSIZE(szBuf2)); // Backup the unfolding text if (ListView_IsInfoTip(plv) && iNewSubHit == 0) { NMLVGETINFOTIP git; git.dwFlags = bItemUnfolded ? LVGIT_UNFOLDED : 0; git.pszText = szBuf; git.cchTextMax = ARRAYSIZE(szBuf); git.iItem = plv->iTTLastHit; git.iSubItem = 0; git.lParam = 0; // for folded items pszText is prepopulated with the // item text, clients should append to this CCSendNotify(&plv->ci, LVN_GETINFOTIP, &git.hdr); if (*szBuf && lstrcmp(szBuf, szBuf2) != 0) { // App changed something - there is a real infotip fInfoTip = TRUE; } } // // Set the margins now before the TTN_SHOW because it will be too late then. // // We want fat margins if we're an infotip, thin margins if we're an // in-place tooltip. // if (fInfoTip) { static const RECT rcMargin = {4, 4, 4, 4}; SendMessage(plv->hwndToolTips, TTM_SETMARGIN, 0, (LPARAM)&rcMargin); CCSetInfoTipWidth(plv->ci.hwnd, plv->hwndToolTips); } else { static const RECT rcMargin = {0, 0, 0, 0}; plv->fPlaceTooltip = TRUE; SendMessage(plv->hwndToolTips, TTM_SETMARGIN, 0, (LPARAM)&rcMargin); CCResetInfoTipWidth(plv->ci.hwnd, plv->hwndToolTips); } Str_Set(&plv->pszTip, pszTip); } } pttt->lpszText = plv->pszTip; // here it is... } break; // Handle custom draw as we want the tooltip painted as a multi-line that // matches the formatting used by the list view. case NM_CUSTOMDRAW: { LPNMTTCUSTOMDRAW pnm = (LPNMTTCUSTOMDRAW) pnmh; if (plv->fPlaceTooltip && (pnm->nmcd.dwDrawStage == CDDS_PREPAINT || pnm->nmcd.dwDrawStage == CDDS_ITEMPREPAINT)) { DWORD dwCustom = 0; // // Set up the customdraw DC to match the font of the LV item. // if (plv->iTTLastHit != -1) { LVFAKEDRAW lvfd; LV_ITEM item; ListView_BeginFakeCustomDraw(plv, &lvfd, &item); item.iItem = plv->iTTLastHit; item.iSubItem = plv->iTTLastSubHit; item.mask = LVIF_PARAM; ListView_OnGetItem(plv, &item); dwCustom = ListView_BeginFakeItemDraw(&lvfd); // If client changed the font, then transfer the font // from our private hdc into the tooltip's HDC. We use // a private HDC because we only want to let the app change // the font, not the colors or anything else. if (dwCustom & CDRF_NEWFONT) { SelectObject(pnm->nmcd.hdc, GetCurrentObject(lvfd.nmcd.nmcd.hdc, OBJ_FONT)); } ListView_EndFakeItemDraw(&lvfd); ListView_EndFakeCustomDraw(&lvfd); } // // The Large Icon tooltip needs to be drawn specially. // if (ListView_IsIconView(plv)) { pnm->uDrawFlags &= ~(DT_SINGLELINE|DT_LEFT); pnm->uDrawFlags |= DT_CENTER|DT_LVWRAP; if (pnm->uDrawFlags & DT_CALCRECT) { pnm->nmcd.rc.right = pnm->nmcd.rc.left + (plv->cxIconSpacing - g_cxLabelMargin * 2); pnm->nmcd.rc.bottom = pnm->nmcd.rc.top + 0x10000; // big number, no limit! } } // Don't return other wacky flags to TT, since all we // did was change the font (if even that) return dwCustom & CDRF_NEWFONT; } } break; case TTN_SHOW: if (plv->iTTLastHit != -1) { if (plv->fPlaceTooltip) { LPNMTTSHOWINFO psi = (LPNMTTSHOWINFO)pnmh; RECT rcLabel; // In case we're doing subitem hit-testing rcLabel.top = plv->iTTLastSubHit; rcLabel.left = LVIR_LABEL; // reposition to allign with the text rect and // set it to topmost if (plv->iTTLastSubHit && ListView_OnGetSubItemRect(plv, plv->iTTLastHit, &rcLabel)) { LV_ITEM item; // we got the subitem rect. When we draw subitems, we give // them SHDT_EXTRAMARGIN, so we have to also rcLabel.left += g_cxLabelMargin * 3; rcLabel.right -= g_cxLabelMargin * 3; // And take the image into account, too. // ListView_OnGetItem will worry about LVS_EX_SUBITEMIMAGES. item.mask = LVIF_IMAGE; item.iImage = -1; item.iItem = plv->iTTLastHit; item.iSubItem = plv->iTTLastSubHit; ListView_OnGetItem(plv, &item); if (item.iImage != -1) rcLabel.left += plv->cxSmIcon; } else { // a tip from subitem zero ListView_GetUnfoldedRect(plv, plv->iTTLastHit, &rcLabel); // SHDrawText actually leaves a g_cxLabelMargin margin rcLabel.left += g_cxLabelMargin; rcLabel.right -= g_cxLabelMargin; } // In report and list views, SHDrawText does vertical // centering (without consulting the custom-draw client, // even, so it just centers by a random amount). if (ListView_IsListView(plv) || ListView_IsReportView(plv)) { rcLabel.top += (rcLabel.bottom - rcLabel.top - plv->cyLabelChar) / 2; } SendMessage(plv->hwndToolTips, TTM_ADJUSTRECT, TRUE, (LPARAM)&rcLabel); MapWindowRect(plv->ci.hwnd, HWND_DESKTOP, &rcLabel); if (!ListView_IsIconView(plv)) { // In non-large-icon view, the label size may be greater than the rect returned by ListView_GetUnfoldedRect. // So don't specify the size SetWindowPos(plv->hwndToolTips, HWND_TOP, rcLabel.left, rcLabel.top, 0, 0, SWP_NOSIZE | SWP_NOACTIVATE | SWP_HIDEWINDOW); } else { SetWindowPos(plv->hwndToolTips, HWND_TOP, rcLabel.left, rcLabel.top, (rcLabel.right - rcLabel.left), (rcLabel.bottom - rcLabel.top), SWP_NOACTIVATE | SWP_HIDEWINDOW); } // This is an inplace tooltip, so disable animation. psi->dwStyle |= TTS_NOANIMATE; return TRUE; } else if (ListView_IsKbdTipTracking(plv)) // Size tip when keyboard tracking { RECT rc; RECT rcTT; RECT rcItem; POINT ptTT; POINT ptItem; MONITORINFO mi = {0}; mi.cbSize = sizeof(MONITORINFO); // Establish item screen position and size ListView_GetItemRect(plv->ci.hwnd, plv->iTracking, &rcItem, LVIR_ICON); ListView_GetItemRect(plv->ci.hwnd, plv->iTracking, &rc, LVIR_BOUNDS); rcItem.top = rc.top; rcItem.bottom = rc.bottom; ptItem.x = rcItem.left; ptItem.y = rcItem.top; ClientToScreen(plv->ci.hwnd, &ptItem); // Get tip rect GetWindowRect(plv->hwndToolTips, &rcTT); // Init tooltip position ptTT.x = ptItem.x + RECTWIDTH(rcItem); ptTT.y = ptItem.y + RECTHEIGHT(rcItem); // Get screen info where tooltip is being displayed GetMonitorInfo(MonitorFromPoint(ptTT, MONITOR_DEFAULTTONEAREST), &mi); // Update tooltip position if it runs off the screen if ((ptTT.x + RECTWIDTH(rcTT)) > mi.rcMonitor.right) ptTT.x = (ptItem.x + g_cxIconMargin) - RECTWIDTH(rcTT); if ((ptTT.y + RECTHEIGHT(rcTT)) > mi.rcMonitor.bottom) ptTT.y = ptItem.y - RECTHEIGHT(rcTT); SetWindowPos(plv->hwndToolTips, NULL, ptTT.x, ptTT.y, 0, 0, SWP_NOSIZE|SWP_NOACTIVATE); return TRUE; } } break; } } return 0; } // Pass the focus to the given window, and then check to see if it exists. // Passing focus can cause the window to be destroyed (by the Explorer // when renaming). BOOL ListView_SetFocus(HWND hwnd) { SetFocus(hwnd); return IsWindow(hwnd); } void ListView_Realize(LV* plv, HDC hdcParam, BOOL fBackground, BOOL fForceRepaint) { if (plv->hpalHalftone) { HDC hdc = hdcParam ? hdcParam : GetDC(plv->ci.hwnd); if (hdc) { BOOL fRepaint; SelectPalette(hdc, plv->hpalHalftone, fBackground); fRepaint = RealizePalette(hdc) || fForceRepaint; if (!hdcParam) ReleaseDC(plv->ci.hwnd, hdc); if (fRepaint) { InvalidateRect(plv->ci.hwnd, NULL, TRUE); } } } } BOOL RectInRect(const RECT* prcOuter, const RECT* prcInner) { RECT rcDummy; return IntersectRect(&rcDummy, prcOuter, prcInner); } LRESULT LVGenerateDragImage(LV* plv, SHDRAGIMAGE* pshdi) { LRESULT lRet = 0; int iNumSelected = plv->nSelected; int iIndex; int iSelectedItem; RECT rc = {0, 0, 0, 0}; RECT rcVisRect; HBITMAP hbmpOld = NULL; HDC hdcDragImage; BOOL fBorderSelect = (plv->exStyle & LVS_EX_BORDERSELECT); // First loop through can get the selection rect if (ListView_IsOwnerData(plv)) { plv->plvrangeSel->lpVtbl->CountIncluded(plv->plvrangeSel, &iNumSelected); } if (iNumSelected == 0) return FALSE; plv->flags |= LVF_DRAGIMAGE; GetClientRect(plv->ci.hwnd, &rcVisRect); // Loop Through and calculate the enclosing rect. for (iIndex = iNumSelected - 1, iSelectedItem = -1; iIndex >= 0; iIndex--) { iSelectedItem = ListView_OnGetNextItem(plv, iSelectedItem, LVNI_SELECTED); if (iSelectedItem != -1) { RECT rcItemBounds; // Make sure this is in the visible region if (ListView_GetItemRect(plv->ci.hwnd, iSelectedItem, &rcItemBounds, LVIR_SELECTBOUNDS) && RectInRect(&rcVisRect, &rcItemBounds)) { UnionRect(&rc, &rc, &rcItemBounds); } } } hdcDragImage = CreateCompatibleDC(NULL); if (hdcDragImage) { RGBQUAD* prgbBits; BITMAPINFO bi; // Need to turn this off because it doesn't look good. plv->exStyle &= ~LVS_EX_BORDERSELECT; // After this rc contains the bounds of all the items in Client Coordinates. // // Mirror the the DC, if the listview is mirrored. // if (plv->ci.dwExStyle & RTL_MIRRORED_WINDOW) { SET_DC_RTL_MIRRORED(hdcDragImage); } #define MAX_DRAG_RECT_WIDTH 300 #define MAX_DRAG_RECT_HEIGHT 300 // If this rect is too big, fix it. if (RECTWIDTH(rc) > MAX_DRAG_RECT_WIDTH) { int iLeft = MAX_DRAG_RECT_WIDTH / 2; int iRight = MAX_DRAG_RECT_WIDTH /2; int iRectOriginalLeft = rc.left; // Is the left boundry outside the visible rect? if (rc.left < plv->ptCapture.x - iLeft) { // Yes, then we have to clip it. rc.left = plv->ptCapture.x - iLeft; } else { // No? Well then shift the visible rect to the right, so that we have // more room. iRight += rc.left - (plv->ptCapture.x - iLeft); } // Is the right boundry outside the visible rect? if (rc.right > plv->ptCapture.x + iRight) { // Yes, then we have to clip it. rc.right = plv->ptCapture.x + iRight; } else { // No? Then try and add it to the left if (rc.left > iRectOriginalLeft) { rc.left -= iRight - (rc.right - plv->ptCapture.x); if (rc.left < iRectOriginalLeft) rc.left = iRectOriginalLeft; } } } if (RECTHEIGHT(rc) > MAX_DRAG_RECT_HEIGHT) { // same for top and bottom: // Is the top boundry outside the visible rect? int iTop = MAX_DRAG_RECT_HEIGHT / 2; int iBottom = MAX_DRAG_RECT_HEIGHT /2; int iRectOriginalTop = rc.top; if (rc.top < plv->ptCapture.y - iTop) { // Yes, then we have to clip it. rc.top = plv->ptCapture.y - iTop; } else { // No? Well then shift the visible rect to the right, so that we have // more room. iBottom += rc.top - (plv->ptCapture.y - iTop); } // Is the right boundry outside the visible rect? if (rc.bottom > plv->ptCapture.y + iBottom) { // Yes, then we have to clip it. rc.bottom = plv->ptCapture.y + iBottom; } else { // No? Then try and add it to the top if (rc.top > iRectOriginalTop) { rc.top -= iBottom - (rc.bottom - plv->ptCapture.y); if (rc.top < iRectOriginalTop) rc.top = iRectOriginalTop; } } } pshdi->sizeDragImage.cx = RECTWIDTH(rc) + 1; pshdi->sizeDragImage.cy = RECTHEIGHT(rc) + 1; bi.bmiHeader.biSize = sizeof(bi.bmiHeader); bi.bmiHeader.biWidth = pshdi->sizeDragImage.cx; bi.bmiHeader.biHeight = pshdi->sizeDragImage.cy; bi.bmiHeader.biPlanes = 1; bi.bmiHeader.biBitCount = 32; bi.bmiHeader.biCompression = BI_RGB; pshdi->hbmpDragImage = CreateDIBSection(hdcDragImage, &bi, DIB_RGB_COLORS, &prgbBits, NULL, 0); if (pshdi->hbmpDragImage) { int iTotal = bi.bmiHeader.biWidth * bi.bmiHeader.biHeight; LVDRAWITEM lvdi; int cItem; RECT rcImage = {0, 0, pshdi->sizeDragImage.cx, pshdi->sizeDragImage.cy}; hbmpOld = SelectObject(hdcDragImage, pshdi->hbmpDragImage); ZeroMemory(prgbBits, pshdi->sizeDragImage.cx * pshdi->sizeDragImage.cy); pshdi->crColorKey = CLR_NONE; // Calculate the offset... The cursor should be in the bitmap rect. if (plv->ci.dwExStyle & RTL_MIRRORED_WINDOW) pshdi->ptOffset.x = rc.right - plv->ptCapture.x; else pshdi->ptOffset.x = plv->ptCapture.x - rc.left; pshdi->ptOffset.y = plv->ptCapture.y - rc.top; lvdi.prcClip = NULL; lvdi.plv = plv; lvdi.nmcd.nmcd.hdc = hdcDragImage; lvdi.pitem = NULL; cItem = ListView_Count(plv); // Now loop through again for the paint cycle for (iIndex = cItem - 1, iSelectedItem = -1; iIndex >= 0; iIndex--) { if (ListView_IsOwnerData(plv)) { iSelectedItem++; plv->plvrangeSel->lpVtbl->NextSelected(plv->plvrangeSel, iSelectedItem, &iSelectedItem); } else { LISTITEM* pitem; iSelectedItem = (int)(UINT_PTR)DPA_FastGetPtr(plv->hdpaZOrder, iIndex); pitem = ListView_FastGetItemPtr(plv, iSelectedItem); if (!(pitem->state & LVIS_SELECTED)) iSelectedItem = -1; } if (iSelectedItem != -1) { int iOldItemDrawing; COLORREF crSave; POINT ptOrigin = {-rc.left, -rc.top}; //Offset the rects by... RECT rcItemBounds; RECT rcTemp; iOldItemDrawing = plv->iItemDrawing; plv->iItemDrawing = iSelectedItem; lvdi.nmcd.nmcd.dwItemSpec = iSelectedItem; ListView_GetRects(plv, iSelectedItem, QUERY_DEFAULT, NULL, NULL, &rcItemBounds, NULL); // Make sure this is in the visible region if (IntersectRect(&rcTemp, &rcVisRect, &rcItemBounds)) { ptOrigin.x += rcItemBounds.left; ptOrigin.y += rcItemBounds.top; // these may get changed lvdi.lpptOrg = &ptOrigin; lvdi.flags = LVDI_NOEFFECTS; lvdi.nmcd.clrText = plv->clrText; lvdi.nmcd.clrTextBk = plv->clrTextBk; lvdi.nmcd.clrFace = plv->clrBk; lvdi.nmcd.iIconEffect = ILD_NORMAL; lvdi.nmcd.iIconPhase = 0; // Save the Background color! crSave = plv->clrBk; plv->clrBk = CLR_NONE; // None so that it "bleeds" into the alpha channel ListView_DrawItem(&lvdi); plv->clrBk = crSave; } plv->iItemDrawing = iOldItemDrawing; } } for (iIndex = 0; iIndex < iTotal; iIndex++) { RGBQUAD* prgb = &prgbBits[iIndex]; if (prgb->rgbReserved == 0 && (prgb->rgbRed || prgb->rgbGreen || prgb->rgbBlue)) // Do we have color an no alpha? { prgb->rgbReserved = 0xFF; } } SelectObject(hdcDragImage, hbmpOld); DeleteDC(hdcDragImage); // We're passing back the created HBMP. lRet = 1; } if (fBorderSelect) plv->exStyle |= LVS_EX_BORDERSELECT; } plv->flags &= ~LVF_DRAGIMAGE; return lRet; } LRESULT ListView_OnEnableGroupView(LV* plv, BOOL fEnable) { if (plv->ci.style & LVS_OWNERDATA) // Not supported in ownerdata case. return -1; if (fEnable ^ plv->fGroupView) { if (fEnable) { // Turning on groupview, so nuke insertmark, because that's not allowed // in group view LVINSERTMARK lvim = {0}; lvim.cbSize = sizeof(LVINSERTMARK); lvim.iItem = -1; ListView_OnSetInsertMark(plv, &lvim); } plv->fGroupView = fEnable; if (fEnable) { if (plv->hdpaGroups == NULL) plv->hdpaGroups = DPA_Create(4); if (plv->hdpaGroups == NULL) return -1; } plv->rcView.left = RECOMPUTE; SetWindowLongPtr(plv->ci.hwnd, GWL_STYLE, GetWindowLongPtr(plv->ci.hwnd, GWL_STYLE) | LVS_AUTOARRANGE); _ListView_RecomputeEx(plv, NULL, 0, TRUE); InvalidateRect(plv->ci.hwnd, NULL, TRUE); ListView_UpdateScrollBars(plv); return 1; } return 0; } LRESULT ListView_SetViewType(LV* plv, WORD wView) { if (wView > LV_VIEW_MAX) return -1; else if (plv->wView != wView) { int i; WORD wViewOld = plv->wView; ListView_DismissEdit(plv, FALSE); // (dli) Setting the small icon width here and only in the case when we go // from large icon view to some other view because of three reasons: // 1. According to chee, we want to set this before we change the style bit in // plv or after we scale. // 2. We don't want to do it after we scale because we want to set the width to // the maximum value so that the items in this listview do not cover each other // 3. we do it from large icon view because large icon view has fixed width for // each item, small icon view width can be scaled. if (wViewOld == LV_VIEW_ICON) ListView_ISetColumnWidth(plv, 0, LV_GetNewColWidth(plv, 0, ListView_Count(plv)-1), FALSE); if (wView == LV_VIEW_TILE) { ListView_RecalcTileSize(plv); } plv->wView = wView; ListView_TypeChange(plv, wViewOld, BOOLIFY(plv->ci.style & LVS_OWNERDRAWFIXED)); // Else we would like to make the most important item to still // be visible. So first we will look for a cursorered item // if this fails, we will look for the first selected item, // else we will simply ask for the first item (assuming the // count > 0 // // And make sure the scrollbars are up to date Note this // also updates some variables that some views need ListView_UpdateScrollBars(plv); i = (plv->iFocus >= 0) ? plv->iFocus : ListView_OnGetNextItem(plv, -1, LVNI_SELECTED); if ((i == -1) && (ListView_Count(plv) > 0)) i = 0; if (i != -1) ListView_OnEnsureVisible(plv, i, TRUE); RedrawWindow(plv->ci.hwnd, NULL, NULL, RDW_INVALIDATE | RDW_ERASE); // Change of styles also changes tooltip policy, so pop it ListView_PopBubble(plv); } return 1; } BOOL ListView_OnGetFrozenSlot(LV* plv, LPRECT pSlotRect) { int cSlots, iWidth = 0, iHeight = 0; LISTITEM *pItem; if((plv->iFrozenSlot == LV_NOFROZENSLOT) || !ListView_IsIconView(plv) || ListView_IsOwnerData(plv) || (pSlotRect == NULL)) //Supported only in Large Icon mode! return FALSE; cSlots = ListView_GetSlotCount(plv, TRUE, &iWidth, &iHeight); //We need to have a valid pItem to pass to ListView_CalcSlotRect() function. pItem = plv->pFrozenItem; //Try to use a frozen item, if present. if(pItem == NULL) pItem = ListView_GetItemPtr(plv, 0); //Or else, use the first item. if(pItem == NULL) //If we couldn't get any pItem, then we can't call CalcSlotRect(). return FALSE; //... Hence, we have to return failure. else { ListView_CalcSlotRect(plv, pItem, plv->iFrozenSlot, cSlots, FALSE, iWidth, iHeight,pSlotRect); return TRUE; } } BOOL ListView_OnSetFrozenSlot(LV* plv, BOOL fFreeze, LPPOINT pPt) { if(!ListView_IsIconView(plv) || ListView_IsOwnerData(plv)) //Supported only in Large Icon mode! return FALSE; if(fFreeze) { //First, find the slot where the given point lies. int cSlots, iWidth = 0, iHeight = 0; cSlots = ListView_GetSlotCount(plv, TRUE, &iWidth, &iHeight); plv->iFrozenSlot = ListView_CalcHitSlot(plv, *pPt, cSlots, iWidth, iHeight); } else { //Unfreeze a frozen slot. plv->iFrozenSlot = LV_NOFROZENSLOT; //No slot is frozen. } return TRUE; } int ListView_OnGetFrozenItem(LV* plv) { int i; LISTITEM *pItem; if((plv->pFrozenItem == NULL) || !ListView_IsIconView(plv) || ListView_IsOwnerData(plv)) //Supported only in Large Icon mode! return LV_NOFROZENITEM; for(i = 0; i < ListView_Count(plv); i++) { pItem = ListView_GetItemPtr(plv, i); if((pItem != NULL) && (pItem == plv->pFrozenItem)) return (i); } return LV_NOFROZENITEM; } BOOL ListView_OnSetFrozenItem(LV* plv, BOOL fFreeze, int iIndex) { LISTITEM *pitem; if(!ListView_IsIconView(plv) || ListView_IsOwnerData(plv)) //Supported only in Large Icon mode! return FALSE; if(fFreeze) { //Freeze the given item. pitem = ListView_GetItemPtr(plv, iIndex); if(pitem == NULL) return FALSE; plv->pFrozenItem = pitem; } else { //Unfreeze the currently frozen item. plv->pFrozenItem = NULL; } return TRUE; } // Item focus changed via the keyboard, start tracking tooltip timeout for keyboard nav popups // BOOL ListView_OnKeyboardSelected(LV* plv, int iNewFocus) { if (iNewFocus >= 0 && plv->hwndToolTips) { // Focus via the keyboard (already cancelled via entry into this function) plv->iTracking = iNewFocus; // Delay will be replaced with an SPI SetTimer(plv->ci.hwnd, IDT_TRACKINGTIP, GetDoubleClickTime() * 2, NULL); } return TRUE; } LRESULT ListView_OnMapIndexToID(LV* plv, UINT iItem) { LISTITEM* pitem; if (!ListView_IsValidItemNumber(plv, iItem) || ListView_IsOwnerData(plv)) { return -1; } pitem = ListView_FastGetItemPtr(plv, iItem); ASSERT (pitem); return (LRESULT)pitem->dwId; } #ifdef DEBUG UINT uAverageSeekCount = 0; UINT uTotalSeeks = 0; UINT uPerSeekCount = 0; #endif LRESULT ListView_OnMapIdToIndex(LV* plv, UINT Id) { DWORD dwRet = -1; UINT cCounter = 0; UINT cItems = ListView_Count(plv); UINT i; if (ListView_IsOwnerData(plv)) return -1; if (plv->iLastId >= cItems) plv->iLastId = 0; DEBUG_CODE(uTotalSeeks++); for (i = plv->iLastId; cCounter < cItems; cCounter++) { LISTITEM* pitem = ListView_FastGetItemPtr(plv, i); if (pitem->dwId == Id) { if (plv->iLastId > i) plv->iIncrement = -1; else plv->iIncrement = 1; plv->iLastId = dwRet = i; break; } DEBUG_CODE(uPerSeekCount++); i += (DWORD)plv->iIncrement; if (i == -1) // Wrapped around to "Less than zero"? i = cItems - 1; if (i >= cItems) i = 0; } DEBUG_CODE(uAverageSeekCount = uPerSeekCount / uTotalSeeks); return (LRESULT)dwRet; } void ListView_OnSize(LV* plv) { if (plv->hwndToolTips) { TOOLINFO ti; if (ListView_IsLabelTip(plv)) { // A truncated label may have been exposed or vice versa. ListView_InvalidateTTLastHit(plv, plv->iTTLastHit); } ti.cbSize = sizeof(ti); ti.hwnd = plv->ci.hwnd; ti.uId = 0; // Resize the tooltip control so that it covers the entire // area of the window when its parent gets resized. GetClientRect(plv->ci.hwnd, &ti.rect); SendMessage(plv->hwndToolTips, TTM_NEWTOOLRECT, 0, (LPARAM) &ti); } // if we're supposed to center the image, // we need to do a full redraw on each size if ((plv->ulBkImageFlags & LVBKIF_SOURCE_MASK) && (plv->ulBkImageFlags & LVBKIF_STYLE_MASK) == LVBKIF_STYLE_NORMAL && (plv->xOffsetPercent || plv->yOffsetPercent)) { InvalidateRect(plv->ci.hwnd, NULL, TRUE); } } BOOL ListView_OnSetViewMargins(LV* plv, RECT *prc) { if (!IsEqualRect(plv->rcViewMargin, *prc)) { plv->rcViewMargin = *prc; plv->rcView.left = RECOMPUTE; InvalidateRect(plv->ci.hwnd, NULL, TRUE); } return TRUE; } BOOL ListView_OnGetViewMargins(LV* plv, RECT *prc) { *prc = plv->rcViewMargin; return TRUE; } LRESULT CALLBACK ListView_WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { LV* plv = ListView_GetPtr(hwnd); if (plv == NULL) { if (uMsg == WM_NCCREATE) { plv = (LV*)LocalAlloc(LPTR, sizeof(LV)); if (!plv) { TraceMsg(TF_ERROR, "ListView: Out of memory"); return 0L; // fail the window create } plv->ci.hwnd = hwnd; plv->flags = LVF_REDRAW; // assume that redrawing enabled! plv->iFocus = -1; // no focus plv->iMark = -1; plv->iSelCol = -1; plv->iDropHilite = -1; // Assume no item has drop hilite... plv->cyItem = plv->cyItemSave = 1; // never let these be zero, not even for a moment plv->hTheme = OpenThemeData(hwnd, L"ListView"); plv->iInsertItem = -1; // No insert mark by default of course plv->clrim = CLR_DEFAULT; plv->iTracking = LVKTT_NOTRACK; plv->hheap = GetProcessHeap(); plv->iFrozenSlot = LV_NOFROZENSLOT; //No slot is frozen to begin with! plv->iIncrement = -1; ListView_SetPtr(hwnd, plv); } goto DoDefault; } if ((uMsg >= WM_MOUSEFIRST) && (uMsg <= WM_MOUSELAST)) { if (plv->exStyle & (LVS_EX_TRACKSELECT|LVS_EX_ONECLICKACTIVATE|LVS_EX_TWOCLICKACTIVATE)) { TRACKMOUSEEVENT tme; tme.cbSize = sizeof(tme); tme.hwndTrack = plv->ci.hwnd; tme.dwHoverTime = plv->dwHoverTime; tme.dwFlags = TME_LEAVE | TME_HOVER | TME_QUERY; // see what's set TrackMouseEvent(&tme); tme.dwFlags &= TME_HOVER | TME_LEAVE; // set these bits if they aren't already set tme.dwFlags ^= TME_LEAVE; if (plv->exStyle & LVS_EX_TRACKSELECT) { tme.dwFlags ^= TME_HOVER; } tme.cbSize = sizeof(tme); tme.hwndTrack = plv->ci.hwnd; tme.dwHoverTime = plv->dwHoverTime; // set it if there's anything to set if (tme.dwFlags & (TME_HOVER | TME_LEAVE)) { TrackMouseEvent(&tme); } } } if (uMsg == g_uDragImages) { return LVGenerateDragImage(plv, (SHDRAGIMAGE*)lParam); } switch (uMsg) { HANDLE_MSG(plv, WM_CREATE, ListView_OnCreate); HANDLE_MSG(plv, WM_DESTROY, ListView_OnDestroy); HANDLE_MSG(plv, WM_ERASEBKGND, ListView_OnEraseBkgnd); HANDLE_MSG(plv, WM_COMMAND, ListView_OnCommand); HANDLE_MSG(plv, WM_SETFOCUS, ListView_OnSetFocus); HANDLE_MSG(plv, WM_KILLFOCUS, ListView_OnKillFocus); HANDLE_MSG(plv, WM_HSCROLL, ListView_OnHScroll); HANDLE_MSG(plv, WM_VSCROLL, ListView_OnVScroll); HANDLE_MSG(plv, WM_GETDLGCODE, ListView_OnGetDlgCode); HANDLE_MSG(plv, WM_SETFONT, ListView_OnSetFont); HANDLE_MSG(plv, WM_GETFONT, ListView_OnGetFont); HANDLE_MSG(plv, WM_TIMER, ListView_OnTimer); HANDLE_MSG(plv, WM_SETREDRAW, ListView_OnSetRedraw); HANDLE_MSG(plv, WM_NCDESTROY, ListView_OnNCDestroy); case WM_SETCURSOR: if (ListView_OnSetCursorMsg(plv)) return TRUE; break; case WM_PALETTECHANGED: if ((HWND)wParam == hwnd) break; case WM_QUERYNEWPALETTE: // Want to pass FALSE if WM_QUERYNEWPALETTE... ListView_Realize(plv, NULL, uMsg == WM_PALETTECHANGED, uMsg == WM_PALETTECHANGED); return TRUE; case LVMP_WINDOWPOSCHANGED: case WM_WINDOWPOSCHANGED: HANDLE_WM_WINDOWPOSCHANGED(plv, wParam, lParam, ListView_OnWindowPosChanged); break; case WM_WINDOWPOSCHANGING: { WINDOWPOS* wp = (WINDOWPOS*)lParam; if ((wp->flags & SWP_SHOWWINDOW)|| (wp->flags & SWP_HIDEWINDOW)) { BOOL fShow = (wp->flags & SWP_SHOWWINDOW); LV_OnShowWindow(plv, fShow); } if (ListView_IsWatermarked(plv)) { RECT rc = {wp->x, wp->y, wp->x + wp->cx, wp->y + wp->y}; // Invalidate New. rc.left = rc.right - plv->szWatermark.cx; rc.top = rc.bottom - plv->szWatermark.cy; InvalidateRect(plv->ci.hwnd, &rc, TRUE); // and Old: GetClientRect(plv->ci.hwnd, &rc); rc.left = rc.right - plv->szWatermark.cx; rc.top = rc.bottom - plv->szWatermark.cy; InvalidateRect(plv->ci.hwnd, &rc, TRUE); } } break; case WM_MBUTTONDOWN: if (ListView_SetFocus(hwnd) && plv->hwndToolTips) RelayToToolTips(plv->hwndToolTips, hwnd, uMsg, wParam, lParam); break; case WM_LBUTTONDBLCLK: case WM_RBUTTONDBLCLK: // Cancel manual tip track on any mouse button down ListView_CancelTipTrack(plv); if (plv->hwndToolTips) RelayToToolTips(plv->hwndToolTips, hwnd, uMsg, wParam, lParam); ListView_OnButtonDown(plv, TRUE, GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam), (UINT) wParam); break; case WM_LBUTTONDOWN: case WM_RBUTTONDOWN: // Cancel manual tip track on any mouse button down ListView_CancelTipTrack(plv); if (plv->hwndToolTips) RelayToToolTips(plv->hwndToolTips, hwnd, uMsg, wParam, lParam); ListView_OnButtonDown(plv, FALSE, GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam), (UINT) wParam); break; case WM_LBUTTONUP: case WM_RBUTTONUP: case WM_MBUTTONUP: case WM_NCMOUSEMOVE: if (plv->hwndToolTips) RelayToToolTips(plv->hwndToolTips, hwnd, uMsg, wParam, lParam); break; case WM_PRINTCLIENT: case WM_PAINT: ListView_OnPaint(plv, (HDC)wParam); return 0; case WM_NCPAINT: { if (plv->hTheme && plv->ci.dwExStyle & WS_EX_CLIENTEDGE) { HRGN hrgn = (wParam != 1) ? (HRGN)wParam : NULL; if (CCDrawNonClientTheme(plv->hTheme, hwnd, hrgn, plv->hbrBk, 0, 0)) { return 1; } } } break; case WM_SHOWWINDOW: LV_OnShowWindow(plv, BOOLFROMPTR(wParam)); break; case WM_MOUSEHOVER: ListView_OnMouseHover(plv, GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam), (UINT) wParam); break; case WM_MOUSELEAVE: ListView_OnSetHotItem(plv, -1); plv->iNoHover = -1; break; case WM_KEYUP: plv->iScrollCount = 0; break; case WM_KEYDOWN: HANDLE_WM_KEYDOWN(plv, wParam, lParam, ListView_OnKey); break; case WM_IME_COMPOSITION: // Now only Korean version is interested in incremental search with composition string. if (g_fDBCSInputEnabled) { if (((ULONG_PTR)GetKeyboardLayout(0L) & 0xF000FFFFL) == 0xE0000412L) { if (ListView_OnImeComposition(plv, wParam, lParam)) { lParam &= ~GCS_RESULTSTR; break; } else return 0; } } break; case WM_CHAR: if (plv->iPuntChar) { plv->iPuntChar--; return TRUE; } else { return HANDLE_WM_CHAR(plv, wParam, lParam, ListView_OnChar); } case WM_WININICHANGE: ListView_OnWinIniChange(plv, wParam, lParam); break; case WM_NOTIFYFORMAT: return CIHandleNotifyFormat(&plv->ci, lParam); case WM_ENABLE: // HACK: we don't get WM_STYLECHANGE on EnableWindow() ListView_EnableWindow(plv, BOOLFROMPTR(wParam)); break; case WM_SYSCOLORCHANGE: InitGlobalColors(); if (plv->ci.style & WS_DISABLED) { if (!(plv->flags & LVF_USERBKCLR)) plv->clrBkSave = g_clrWindow; ListView_OnSetBkColor(plv, g_clrBtnFace); } else if (!(plv->flags & LVF_USERBKCLR)) { ListView_OnSetBkColor(plv, g_clrWindow); } if (plv->exStyle & LVS_EX_CHECKBOXES) { ListView_InitCheckBoxes(plv, FALSE); } plv->crHeader = GetSysColor(COLOR_WINDOWTEXT); plv->crTop = GetSysColor(COLOR_BTNFACE); plv->crLeft = GetSysColor(COLOR_BTNFACE); // 98/11/19 #249967 vtan: Always invalidate the list view // rectangle so that the color change causes a refresh. InvalidateRect(plv->ci.hwnd, NULL, TRUE); break; // don't use HANDLE_MSG because this needs to go to the default handler case WM_SYSKEYDOWN: HANDLE_WM_SYSKEYDOWN(plv, wParam, lParam, ListView_OnKey); break; case WM_UPDATEUISTATE: { DWORD dwUIStateMask = MAKEWPARAM(0xFFFF, UISF_HIDEFOCUS); // we care only about focus not accel, and redraw only if changed if (CCOnUIState(&(plv->ci), WM_UPDATEUISTATE, wParam & dwUIStateMask, lParam)) { if (plv->iFocus >= 0) { // an item has the focus, invalidate it ListView_InvalidateItem(plv, plv->iFocus, FALSE, RDW_INVALIDATE | RDW_ERASE); } } goto DoDefault; } case LVM_GETITEMA: return (LRESULT)ListView_OnGetItemA(plv, (LV_ITEMA *)lParam); case LVM_SETITEMA: return (LRESULT)ListView_OnSetItemA(plv, (LV_ITEMA *)lParam); case LVM_INSERTITEMA: return (LRESULT)ListView_OnInsertItemA(plv, (LV_ITEMA *)lParam); case LVM_FINDITEMA: return (LRESULT)ListView_OnFindItemA(plv, (int)wParam, (LV_FINDINFOA *)lParam); case LVM_GETSTRINGWIDTHA: return (LRESULT)ListView_OnGetStringWidthA(plv, (LPCSTR)lParam, NULL); case LVM_GETCOLUMNA: return (LRESULT)ListView_OnGetColumnA(plv, (int)wParam, (LV_COLUMNA *)lParam); case LVM_SETCOLUMNA: return (LRESULT)ListView_OnSetColumnA(plv, (int)wParam, (LV_COLUMNA *)lParam); case LVM_INSERTCOLUMNA: return (LRESULT)ListView_OnInsertColumnA(plv, (int)wParam, (LV_COLUMNA *)lParam); case LVM_GETITEMTEXTA: return (LRESULT)ListView_OnGetItemTextA(plv, (int)wParam, (LV_ITEMA *)lParam); case LVM_SETITEMTEXTA: if (!lParam) return FALSE; return (LRESULT)ListView_OnSetItemTextA(plv, (int)wParam, ((LV_ITEMA *)lParam)->iSubItem, (LPCSTR)((LV_ITEMA *)lParam)->pszText); case LVM_GETBKIMAGEA: return (LRESULT)ListView_OnGetBkImageA(plv, (LPLVBKIMAGEA)lParam); case LVM_SETBKIMAGEA: return (LRESULT)ListView_OnSetBkImageA(plv, (LPLVBKIMAGEA)lParam); case WM_STYLECHANGING: ListView_OnStyleChanging(plv, (UINT)wParam, (LPSTYLESTRUCT)lParam); return 0; case WM_STYLECHANGED: ListView_OnStyleChanged(plv, (UINT) wParam, (LPSTYLESTRUCT)lParam); return 0L; case WM_HELP: return ListView_OnHelp(plv, (LPHELPINFO)lParam); case LVM_GETIMAGELIST: return (LRESULT)(UINT_PTR)(ListView_OnGetImageList(plv, (int)wParam)); case LVM_SETIMAGELIST: return (LRESULT)(UINT_PTR)ListView_OnSetImageList(plv, (HIMAGELIST)lParam, (int)wParam); case LVM_GETBKCOLOR: return (LRESULT)(plv->ci.style & WS_DISABLED ? plv->clrBkSave : plv->clrBk); case LVM_SETBKCOLOR: plv->flags |= LVF_USERBKCLR; if (plv->ci.style & WS_DISABLED) { plv->clrBkSave = (COLORREF)lParam; return TRUE; } else { return (LRESULT)ListView_OnSetBkColor(plv, (COLORREF)lParam); } case LVM_GETTEXTCOLOR: return (LRESULT)plv->clrText; case LVM_SETTEXTCOLOR: plv->clrText = (COLORREF)lParam; return TRUE; case LVM_GETTEXTBKCOLOR: return (LRESULT)plv->clrTextBk; case LVM_SETTEXTBKCOLOR: plv->clrTextBk = (COLORREF)lParam; return TRUE; case LVM_GETHOTLIGHTCOLOR: return (LRESULT)plv->clrHotlight; case LVM_SETHOTLIGHTCOLOR: plv->clrHotlight = (COLORREF)lParam; return TRUE; case LVM_GETITEMCOUNT: if (ListView_IsOwnerData(plv)) return (LRESULT)plv->cTotalItems; else if (!plv->hdpa) return 0; else return (LRESULT)DPA_GetPtrCount(plv->hdpa); break; case LVM_GETITEM: return (LRESULT)ListView_OnGetItem(plv, (LV_ITEM*)lParam); case LVM_GETITEMSTATE: return (LRESULT)ListView_OnGetItemState(plv, (int)wParam, (UINT)lParam); case LVM_SETITEMSTATE: if (!lParam) return FALSE; return (LRESULT)ListView_OnSetItemState(plv, (int)wParam, ((LV_ITEM *)lParam)->state, ((LV_ITEM *)lParam)->stateMask); case LVM_SETITEMTEXT: if (!lParam) return FALSE; return (LRESULT)ListView_OnSetItemText(plv, (int)wParam, ((LV_ITEM *)lParam)->iSubItem, (LPCTSTR)((LV_ITEM *)lParam)->pszText); case LVM_GETITEMTEXT: return (LRESULT)ListView_OnGetItemText(plv, (int)wParam, (LV_ITEM *)lParam); case LVM_GETBKIMAGE: return (LRESULT)ListView_OnGetBkImage(plv, (LPLVBKIMAGE)lParam); case LVM_SETBKIMAGE: return (LRESULT)ListView_OnSetBkImage(plv, (LPLVBKIMAGE)lParam); case LVM_GETSELECTEDCOLUMN: return plv->iLastColSort; case LVM_SETSELECTEDCOLUMN: plv->iLastColSort = (int) wParam; if (ListView_IsTileView(plv)) { // Tileview displays the selected column on the second line, if available. The second // line might be blank w/o it. So when this changes, we need to recompute each tile. if (!ListView_IsOwnerData(plv)) { int i; for (i = 0; i < ListView_Count(plv); i++) { LISTITEM *pitem = ListView_FastGetItemPtr(plv, i); ListView_SetSRecompute(pitem); } } plv->rcView.left = RECOMPUTE; InvalidateRect(plv->ci.hwnd, NULL, TRUE); } return 1; case LVM_SETVIEW: return ListView_SetViewType(plv, (WORD)wParam); case LVM_GETVIEW: return plv->wView; case LVM_SETITEM: return (LRESULT)ListView_OnSetItem(plv, (const LV_ITEM*)lParam); case LVM_INSERTITEM: return (LRESULT)ListView_OnInsertItem(plv, (const LV_ITEM*)lParam); case LVM_DELETEITEM: return (LRESULT)ListView_OnDeleteItem(plv, (int)wParam); case LVM_UPDATE: ListView_OnUpdate(plv, (int)wParam); UpdateWindow(plv->ci.hwnd); return TRUE; case LVM_DELETEALLITEMS: lParam = (LRESULT)ListView_OnDeleteAllItems(plv); // Optimization: Instead of sending out a zillion EVENT_OBJECT_DESTROY's, // we send out a destroy of ourselves followed by a fresh create. // For compatibility with IE4, we still send out the REORDER notification. NotifyWinEvent(EVENT_OBJECT_REORDER, hwnd, OBJID_CLIENT, 0); ListView_NotifyRecreate(plv); return lParam; case LVM_GETITEMRECT: return (LRESULT)ListView_OnGetItemRect(plv, (int)wParam, (RECT*)lParam); case LVM_GETSUBITEMRECT: return (LRESULT)ListView_OnGetSubItemRect(plv, (int)wParam, (LPRECT)lParam); case LVM_SUBITEMHITTEST: return (LRESULT)ListView_OnSubItemHitTest(plv, (LPLVHITTESTINFO)lParam); case LVM_GETISEARCHSTRINGA: if (GetFocus() == plv->ci.hwnd) return (LRESULT)GetIncrementSearchStringA(&plv->is, plv->ci.uiCodePage, (LPSTR)lParam); else return 0; case LVM_GETISEARCHSTRING: if (GetFocus() == plv->ci.hwnd) return (LRESULT)GetIncrementSearchString(&plv->is, (LPTSTR)lParam); else return 0; case LVM_GETITEMSPACING: if (wParam) return MAKELONG(plv->cxItem, plv->cyItem); else return MAKELONG(plv->cxIconSpacing, plv->cyIconSpacing); case LVM_GETNEXTITEM: return (LRESULT)ListView_OnGetNextItem(plv, (int)wParam, (UINT)lParam); case LVM_FINDITEM: return (LRESULT)ListView_OnFindItem(plv, (int)wParam, (const LV_FINDINFO*)lParam); case LVM_SETSELECTIONMARK: { int iOldMark = plv->iMark; int iNewMark = (int)lParam; if (iNewMark == -1 || ListView_IsValidItemNumber(plv, iNewMark)) { plv->iMark = iNewMark; } return iOldMark; } case LVM_GETSELECTIONMARK: return plv->iMark; case LVM_GETITEMPOSITION: return (LRESULT)ListView_OnGetItemPosition(plv, (int)wParam, (POINT*)lParam); case LVM_SETITEMPOSITION: return (LRESULT)ListView_OnSetItemPosition(plv, (int)wParam, GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam)); case LVM_SETITEMPOSITION32: if (!lParam) return FALSE; return (LRESULT)ListView_OnSetItemPosition(plv, (int)wParam, ((LPPOINT)lParam)->x, ((LPPOINT)lParam)->y); case LVM_SCROLL: { int dx = (int)wParam; int dy = (int)lParam; return (LRESULT)(ListView_ValidateScrollParams(plv, &dx, &dy) && ListView_OnScroll(plv, dx, dy)); } case LVM_ENSUREVISIBLE: return (LRESULT)ListView_OnEnsureVisible(plv, (int)wParam, BOOLFROMPTR(lParam)); case LVM_REDRAWITEMS: return (LRESULT)ListView_OnRedrawItems(plv, (int)wParam, (int)lParam); case LVM_ARRANGE: return (LRESULT)ListView_OnArrange(plv, (UINT)wParam); case LVM_GETEDITCONTROL: return (LRESULT)(UINT_PTR)plv->hwndEdit; case LVM_EDITLABELA: { LPWSTR lpEditString = NULL; HWND hRet; if (lParam) { lpEditString = ProduceWFromA(plv->ci.uiCodePage, (LPSTR)lParam); } hRet = ListView_OnEditLabel(plv, (int)wParam, lpEditString); if (lpEditString) { FreeProducedString(lpEditString); } return (LRESULT)hRet; } case LVM_EDITLABEL: return (LRESULT)(UINT_PTR)ListView_OnEditLabel(plv, (int)wParam, (LPTSTR)lParam); case LVM_HITTEST: return (LRESULT)ListView_OnHitTest(plv, (LV_HITTESTINFO*)lParam); case LVM_GETSTRINGWIDTH: return (LRESULT)ListView_OnGetStringWidth(plv, (LPCTSTR)lParam, NULL); case LVM_GETCOLUMN: return (LRESULT)ListView_OnGetColumn(plv, (int)wParam, (LV_COLUMN*)lParam); case LVM_SETCOLUMN: return (LRESULT)ListView_OnSetColumn(plv, (int)wParam, (const LV_COLUMN*)lParam); case LVM_SETCOLUMNORDERARRAY: return SendMessage(plv->hwndHdr, HDM_SETORDERARRAY, wParam, lParam); case LVM_GETCOLUMNORDERARRAY: return SendMessage(plv->hwndHdr, HDM_GETORDERARRAY, wParam, lParam); case LVM_GETHEADER: { HWND hwndOld = plv->hwndHdr; if (lParam && IsWindow((HWND)lParam)) { plv->hwndHdr = (HWND)lParam; } return (LRESULT)hwndOld; } case LVM_INSERTCOLUMN: return (LRESULT)ListView_OnInsertColumn(plv, (int)wParam, (const LV_COLUMN*)lParam); case LVM_DELETECOLUMN: return (LRESULT)ListView_OnDeleteColumn(plv, (int)wParam); case LVM_CREATEDRAGIMAGE: return (LRESULT)(UINT_PTR)ListView_OnCreateDragImage(plv, (int)wParam, (LPPOINT)lParam); case LVMI_PLACEITEMS: if (plv->uUnplaced) { ListView_Recompute(plv); ListView_UpdateScrollBars(plv); } return 0; case LVM_GETVIEWRECT: return (LPARAM)ListView_OnGetViewRect(plv, (RECT*)lParam); case LVM_GETCOLUMNWIDTH: return (LPARAM)ListView_OnGetColumnWidth(plv, (int)wParam); case LVM_SETCOLUMNWIDTH: return (LPARAM)ListView_ISetColumnWidth(plv, (int)wParam, GET_X_LPARAM(lParam), TRUE); case LVM_SETCALLBACKMASK: plv->stateCallbackMask = (UINT)wParam; return (LPARAM)TRUE; case LVM_GETCALLBACKMASK: return (LPARAM)(UINT)plv->stateCallbackMask; case LVM_GETTOPINDEX: return (LPARAM)ListView_OnGetTopIndex(plv); case LVM_GETCOUNTPERPAGE: return (LPARAM)ListView_OnGetCountPerPage(plv); case LVM_GETORIGIN: return (LPARAM)ListView_OnGetOrigin(plv, (POINT*)lParam); case LVM_SETITEMCOUNT: return ListView_OnSetItemCount(plv, (int)wParam, (DWORD)lParam); case LVM_GETSELECTEDCOUNT: if (ListView_IsOwnerData(plv)) { plv->plvrangeSel->lpVtbl->CountIncluded(plv->plvrangeSel, &plv->nSelected); } return plv->nSelected; case LVM_SORTITEMS: return ListView_OnSortItems(plv, (LPARAM)wParam, (PFNLVCOMPARE)lParam, TRUE); case LVM_SORTITEMSEX: return ListView_OnSortItems(plv, (LPARAM)wParam, (PFNLVCOMPARE)lParam, FALSE); case LVM_SETEXTENDEDLISTVIEWSTYLE: return ListView_ExtendedStyleChange(plv, (DWORD) lParam, (DWORD) wParam); case LVM_GETEXTENDEDLISTVIEWSTYLE: return plv->exStyle; case LVM_GETHOVERTIME: return plv->dwHoverTime; case LVM_SETHOVERTIME: { DWORD dwRet = plv->dwHoverTime; plv->dwHoverTime = (DWORD)lParam; return dwRet; } case LVM_GETTOOLTIPS: return (LRESULT)plv->hwndToolTips; case LVM_SETTOOLTIPS: { HWND hwndToolTips = plv->hwndToolTips; plv->hwndToolTips = (HWND)wParam; return (LRESULT)hwndToolTips; } case LVM_SETICONSPACING: { DWORD dwRet = ListView_OnSetIconSpacing(plv, lParam); // rearrange as necessary if (ListView_RedrawEnabled(plv) && (ListView_IsSmallView(plv) || ListView_IsIconView(plv))) { ListView_ArrangeOrSnapToGrid(plv); } return dwRet; } case LVM_SETHOTITEM: { int iOld = plv->iHot; int iNew = (int)wParam; if (iNew == -1 || ListView_IsValidItemNumber(plv, iNew)) { ListView_OnSetHotItem(plv, (int)wParam); } return iOld; } case LVM_GETHOTITEM: return plv->iHot; // hCurHot is used iff LVS_EX_TRACKSELECT case LVM_SETHOTCURSOR: { HCURSOR hCurOld = plv->hCurHot; plv->hCurHot = (HCURSOR)lParam; return (LRESULT)hCurOld; } case LVM_GETHOTCURSOR: if (!plv->hCurHot) { plv->hCurHot = LoadCursor(NULL, IDC_HAND); } return (LRESULT)plv->hCurHot; case LVM_APPROXIMATEVIEWRECT: return ListView_OnApproximateViewRect(plv, (int)wParam, GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam)); case LVM_SETLVRANGEOBJECT: return ListView_OnSetLVRangeObject(plv, (int)wParam, (ILVRange *)lParam); case LVM_SETWORKAREAS: ListView_OnSetWorkAreas(plv, (int)wParam, (RECT *)lParam); return 0; case LVM_GETWORKAREAS: ListView_OnGetWorkAreas(plv, (int)wParam, (RECT *)lParam); return 0; case LVM_GETNUMBEROFWORKAREAS: ListView_OnGetNumberOfWorkAreas(plv, (int *)lParam); return 0; case LVM_RESETEMPTYTEXT: plv->fNoEmptyText = FALSE; Str_Set(&plv->pszEmptyText, NULL); if (ListView_Count(plv) == 0) InvalidateRect(plv->ci.hwnd, NULL, TRUE); return 1; case LVM_INSERTGROUPSORTED: return ListView_OnInsertGroupSorted(plv, (LVINSERTGROUPSORTED*) wParam); case LVM_SORTGROUPS: return ListView_OnSortGroups(plv, (PFNLVGROUPCOMPARE)wParam, (void *)lParam); case LVM_ENABLEGROUPVIEW: return ListView_OnEnableGroupView(plv, (BOOL)wParam); case LVM_ISGROUPVIEWENABLED: return plv->fGroupView; case LVM_INSERTGROUP: return ListView_OnInsertGroup(plv, (int) wParam, (PLVGROUP)lParam); case LVM_SETGROUPINFO: return ListView_OnSetGroupInfo(plv, (int) wParam, (PLVGROUP)lParam); case LVM_GETGROUPINFO: return ListView_OnGetGroupInfo(plv, (int) wParam, (PLVGROUP)lParam); case LVM_REMOVEGROUP: return ListView_OnRemoveGroup(plv, (int) wParam); case LVM_REMOVEALLGROUPS: return ListView_OnRemoveAllGroups(plv); case LVM_HASGROUP: { LISTGROUP* pgrp = ListView_FindGroupFromID(plv, (int)wParam, NULL); if (pgrp) return 1; else return 0; } break; case LVM_SETGROUPMETRICS: return ListView_OnSetGroupMetrics(plv, (PLVGROUPMETRICS)lParam); case LVM_GETGROUPMETRICS: return ListView_OnGetGroupMetrics(plv, (PLVGROUPMETRICS)lParam); case LVM_SETTILEVIEWINFO: return ListView_OnSetTileViewInfo(plv, (PLVTILEVIEWINFO)lParam); case LVM_GETTILEVIEWINFO: return ListView_OnGetTileViewInfo(plv, (PLVTILEVIEWINFO)lParam); case LVM_SETTILEINFO: return ListView_OnSetTileInfo(plv, (PLVTILEINFO)lParam); case LVM_GETTILEINFO: return ListView_OnGetTileInfo(plv, (PLVTILEINFO)lParam); case LVM_SETINSERTMARK: if (ListView_IsRearrangeableView(plv) && (plv->ci.style & LVS_AUTOARRANGE) && !plv->fGroupView) return ListView_OnSetInsertMark(plv, (LPLVINSERTMARK)lParam); else return FALSE; case LVM_GETINSERTMARK: { LPLVINSERTMARK plvim = (LPLVINSERTMARK)lParam; if (plvim->cbSize != sizeof(LVINSERTMARK)) return FALSE; plvim->dwFlags = (plv->fInsertAfter ? LVIM_AFTER : 0) | LVIM_SETFROMINFO; plvim->iItem = plv->iInsertItem; return TRUE; } case LVM_GETINSERTMARKRECT: { return ListView_OnGetInsertMarkRect(plv, (LPRECT)lParam); } case LVM_SETINSERTMARKCOLOR: { LRESULT lres = (LRESULT)ListView_OnGetInsertMarkColor(plv); plv->clrim = (COLORREF) lParam; return lres; } case LVM_GETINSERTMARKCOLOR: return ListView_OnGetInsertMarkColor(plv); case LVM_INSERTMARKHITTEST: { LPPOINT ppt = (LPPOINT)wParam; return ListView_OnInsertMarkHitTest(plv, ppt->x, ppt->y, (LPLVINSERTMARK)lParam); } case LVM_SETINFOTIP: { return ListView_OnSetInfoTip(plv, (PLVSETINFOTIP)lParam); } case LVM_SETOUTLINECOLOR: { LRESULT lres = (LRESULT)plv->clrOutline; plv->clrOutline = (COLORREF) lParam; return lres; } case LVM_GETOUTLINECOLOR: return (LRESULT)plv->clrOutline; case LVM_SETFROZENITEM: return ListView_OnSetFrozenItem(plv, (BOOL) wParam, (int) lParam); case LVM_GETFROZENITEM: return ListView_OnGetFrozenItem(plv); case LVM_SETFROZENSLOT: return ListView_OnSetFrozenSlot(plv, (BOOL) wParam, (LPPOINT)lParam); case LVM_GETFROZENSLOT: return ListView_OnGetFrozenSlot(plv, (LPRECT)lParam); case LVM_SETVIEWMARGINS: return ListView_OnSetViewMargins(plv, (LPRECT)lParam); case LVM_GETVIEWMARGINS: return ListView_OnGetViewMargins(plv, (LPRECT)lParam); case LVM_KEYBOARDSELECTED: ListView_CancelTipTrack(plv); return lParam == 0 ? ListView_OnKeyboardSelected(plv, (int)wParam) : FALSE; case LVM_CANCELEDITLABEL: ListView_DismissEdit(plv, FALSE); return 1; case LVM_MAPINDEXTOID: return ListView_OnMapIndexToID(plv, (UINT)wParam); case LVM_MAPIDTOINDEX: return ListView_OnMapIdToIndex(plv, (UINT)wParam); case LVM_ISITEMVISIBLE: if (ListView_IsValidItemNumber(plv, (UINT)wParam)) { return ListView_IsItemVisibleI(plv, (UINT)wParam); } else { return FALSE; } case WM_SIZE: if (plv) { ListView_OnSize(plv); } break; case WM_NOTIFY: return ListView_OnNotify(plv, wParam, (LPNMHDR)lParam); case WM_MOUSEMOVE: // Cancel manual track if mouse moved if (plv->lLastMMove != lParam) { ListView_CancelTipTrack(plv); if (plv->hwndToolTips) { UINT uFlags; int iHit, iSubHit; RelayToToolTips(plv->hwndToolTips, hwnd, uMsg, wParam, lParam); if (!ListView_IsKbdTipTracking(plv)) { // check that we are still on the hit item, pop it! iHit = _ListView_ItemHitTest(plv, GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam), &uFlags, &iSubHit); if (iHit != plv->iTTLastHit || iSubHit != plv->iTTLastSubHit) ListView_PopBubble(plv); } } } plv->lLastMMove = lParam; ListView_OnMouseMove(plv, GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam), (UINT) wParam); break; case WM_GETOBJECT: if (lParam == OBJID_QUERYCLASSNAMEIDX) return MSAA_CLASSNAMEIDX_LISTVIEW; break; case WM_THEMECHANGED: if (plv->hTheme) CloseThemeData(plv->hTheme); plv->hTheme = OpenThemeData(plv->ci.hwnd, L"ListView"); InvalidateRect(plv->ci.hwnd, NULL, TRUE); break; default: { LRESULT lres; if (CCWndProc(&plv->ci, uMsg, wParam, lParam, &lres)) return lres; } if (uMsg == g_msgMSWheel) { DWORD dwStyle; int sb; SCROLLINFO si; int cScrollUnitsPerLine; int cPage; int cLinesPerDetant; int cDetants; int dPos; int iWheelDelta = (int)(short)HIWORD(wParam); BOOL fScroll = !(wParam & (MK_SHIFT | MK_CONTROL)); BOOL fDataZoom = (BOOL) (wParam & MK_SHIFT); // Update count of scroll amount gcWheelDelta -= iWheelDelta; cDetants = gcWheelDelta / WHEEL_DELTA; if (cDetants != 0) { gcWheelDelta %= WHEEL_DELTA; } if (fScroll) { if (g_ucScrollLines > 0 && cDetants != 0 && ((WS_VSCROLL | WS_HSCROLL) & (dwStyle = ListView_GetWindowStyle(plv)))) { sb = (dwStyle & WS_VSCROLL) ? SB_VERT : SB_HORZ; // Get the scroll amount of one line cScrollUnitsPerLine = _ListView_GetScrollUnitsPerLine(plv, sb); ASSERT(cScrollUnitsPerLine > 0); si.cbSize = sizeof(SCROLLINFO); si.fMask = SIF_PAGE | SIF_POS; if (!ListView_GetScrollInfo(plv, sb, &si)) return 1; // The size of a page is at least one line, and // leaves one line of overlap cPage = (max(cScrollUnitsPerLine, (int)si.nPage - cScrollUnitsPerLine)) / cScrollUnitsPerLine; // Don't scroll more than one page per detant cLinesPerDetant = (int) min((ULONG) cPage, (ULONG) g_ucScrollLines); dPos = cLinesPerDetant * cDetants * cScrollUnitsPerLine; ListView_DismissEdit(plv, FALSE); ListView_ComOnScroll(plv, SB_THUMBTRACK, si.nPos + dPos, sb, cScrollUnitsPerLine, - 1); ListView_UpdateScrollBars(plv); // After scrolling, the tooltip might need to change // so send the tooltip a fake mousemove message to force // a recompute. We use WM_NCMOUSEMOVE since our lParam // is in screen coordinates, not client coordinates. ListView_PopBubble(plv); RelayToToolTips(plv->hwndToolTips, plv->ci.hwnd, WM_NCMOUSEMOVE, HTCLIENT, lParam); } return 1; } else if (fDataZoom) { LV_HITTESTINFO ht; ht.pt.x = GET_X_LPARAM(lParam); ht.pt.y = GET_Y_LPARAM(lParam); ScreenToClient(hwnd, &(ht.pt)); // If we are rolling forward and we hit an item then navigate // into that item (simulate dblclk which will open it). Otherwise // just fall through so it isn't handled. In that case if we // are being hosted in explorer it will do a backwards // history navigation. if ((iWheelDelta > 0) && (ListView_OnSubItemHitTest(plv, &ht) >= 0) && (ht.flags & LVHT_ONITEM) && cDetants != 0) { BYTE aKeyState[256]; // This is a bit yucky but when ListView_HandleMouse sends the // notification to the listview owner we need to make sure that // it doesn't think the shift key is down. Otherwise it may // perform some "alternate" action but in this case we always // want it to perform the default open action. // // Strip the high bit of VK_SHIFT so that the shift key is // not down. GetKeyboardState(aKeyState); aKeyState[VK_SHIFT] &= 0x7f; SetKeyboardState(aKeyState); ListView_HandleMouse(plv, FALSE, ht.pt.x, ht.pt.y, 0, TRUE); ListView_HandleMouse(plv, TRUE, ht.pt.x, ht.pt.y, 0, TRUE); return 1; } // else fall through } } break; } DoDefault: return DefWindowProc(hwnd, uMsg, wParam, lParam); } BOOL Listview_UpdateViewEffects(LV* plv) { BOOL fChanged = FALSE; UINT fScroll = SHRegGetBoolUSValue(REGSTR_EXPLORER_ADVANCED, TEXT("ListviewScrollOver"), FALSE, // Don't ignore HKCU LISTVIEW_VFX_DEFAULT); // Assume a fast enough machine UINT fWatermark = SHRegGetBoolUSValue(REGSTR_EXPLORER_ADVANCED, TEXT("ListviewWatermark"), FALSE, // Don't ignore HKCU LISTVIEW_VFX_DEFAULT); // Assume a fast enough machine UINT fAlphaSelect = SHRegGetBoolUSValue(REGSTR_EXPLORER_ADVANCED, TEXT("ListviewAlphaSelect"), FALSE, // Don't ignore HKCU LISTVIEW_VFX_DEFAULT); // Assume a fast enough machine UINT fShadow = SHRegGetBoolUSValue(REGSTR_EXPLORER_ADVANCED, TEXT("ListviewShadow"), FALSE, // Don't ignore HKCU LISTVIEW_VFX_DEFAULT); // Assume a fast enough machine if (plv->fListviewAlphaSelect != fAlphaSelect || plv->fListviewShadowText != fShadow || plv->fListviewWatermarkBackgroundImages != fScroll || plv->fListviewEnableWatermark != fWatermark) { fChanged = TRUE; } plv->fListviewAlphaSelect = BOOLIFY(fAlphaSelect); plv->fListviewShadowText = BOOLIFY(fShadow); plv->fListviewWatermarkBackgroundImages = BOOLIFY(fScroll); plv->fListviewEnableWatermark = BOOLIFY(fWatermark); return fChanged; } void ListView_OnWinIniChange(LV* plv, WPARAM wParam, LPARAM lParam) { // REARCHITECT: will this also catch sysparametersinfo? // we need a general way of handling this, not // just relying on the listview. InitGlobalMetrics(wParam); if (Listview_UpdateViewEffects(plv)) { InvalidateRect(plv->ci.hwnd, NULL, TRUE); } switch (wParam) { case 0: case SPI_SETNONCLIENTMETRICS: case SPI_SETICONTITLELOGFONT: case SPI_SETICONMETRICS: // If wParam is 0, only reload settings if lParam is 0 too. This catches the wild-card scenario // (like the old plus tab which does WM_WININICHANGE, 0, 0) but allows us to ignore wParam = 0 // and lParam = lpszSectionName. Reduces unecessary flashing. if (wParam || !lParam) { if (!(plv->flags & LVF_ICONSPACESET)) ListView_OnSetIconSpacing(plv, (LPARAM)-1); if (plv->flags & LVF_FONTCREATED) ListView_OnSetFont(plv, NULL, TRUE); // Force a recalc of all the icon regions by stripping and // then adding back the LVS_EX_REGIONAL bit. if (plv->exStyle & LVS_EX_REGIONAL) { ListView_ExtendedStyleChange(plv, 0, LVS_EX_REGIONAL); ListView_ExtendedStyleChange(plv, LVS_EX_REGIONAL, LVS_EX_REGIONAL); } } break; default: break; } // If we are in an Iconic view and the user is in autoarrange mode, // then we need to arrange the items. // if ((ListView_IsSmallView(plv) || ListView_IsIconView(plv))) { // Call off to the arrange function. if (ListView_IsOwnerData(plv)) ListView_OnArrange(plv, LVA_DEFAULT); else ListView_ArrangeOrSnapToGrid(plv); } } BOOL ListView_OnCreate(LV* plv, CREATESTRUCT* lpCreateStruct) { Listview_UpdateViewEffects(plv); InitDitherBrush(); CIInitialize(&plv->ci, plv->ci.hwnd, lpCreateStruct); plv->wView = (WORD)(plv->ci.style & LVS_TYPEMASK); plv->dwExStyle = lpCreateStruct->dwExStyle; if (plv->ci.style & WS_VISIBLE) plv->flags |= LVF_VISIBLE; ListView_GetRegIASetting(&g_bUseDblClickTimer); if (ListView_IsOwnerData(plv)) { // ownerdata initialization plv->plvrangeSel = LVRange_Create(); if (NULL == plv->plvrangeSel) goto error0; plv->plvrangeCut = LVRange_Create(); if (NULL == plv->plvrangeCut) goto error0; } else { ASSERT(plv->plvrangeSel == NULL); ASSERT(plv->plvrangeCut == NULL); plv->hdpa = DPA_CreateEx(LV_HDPA_GROW, plv->hheap); if (!plv->hdpa) goto error0; plv->hdpaZOrder = DPA_CreateEx(LV_HDPA_GROW, plv->hheap); if (!plv->hdpaZOrder) goto error1; } ASSERT(plv->nWorkAreas == 0); ASSERT(plv->prcWorkAreas == NULL); ASSERT(plv->fIconsPositioned == FALSE); plv->iNoHover = -1; plv->dwHoverTime = HOVER_DEFAULT; plv->iHot = -1; plv->iEdit = -1; plv->iFocus = -1; plv->iDrag = -1; plv->iTTLastHit = -1; plv->iFreeSlot = -1; plv->rcView.left = RECOMPUTE; plv->iLastColSort = -1; ASSERT(plv->sizeTile.cx == 0); ASSERT(plv->sizeTile.cy == 0); ASSERT(plv->dwTileFlags == 0); plv->cSubItems = 1; SetRect(&plv->rcBorder, 0, 12, 0, 0); plv->crHeader = GetSysColor(COLOR_WINDOWTEXT); plv->crTop = GetSysColor(COLOR_BTNFACE); plv->crBottom = CLR_NONE; plv->crLeft = CLR_NONE; plv->crRight = CLR_NONE; plv->paddingLeft = 12; plv->paddingTop = 12; plv->paddingRight = 0; plv->paddingBottom = 12; plv->szWatermark.cx = 200; plv->szWatermark.cy = 200; ASSERT(plv->iMSAAMin == plv->iMSAAMax); plv->sizeClient.cx = lpCreateStruct->cx; plv->sizeClient.cy = lpCreateStruct->cy; // Setup flag to say if positions are in small or large view if (ListView_IsSmallView(plv)) plv->flags |= LVF_ICONPOSSML; // force calculation of listview metrics ListView_OnSetFont(plv, NULL, FALSE); plv->cxItem = ListView_ComputeCXItemSize(plv); // if we're in ownerdraw report mode, the size got saved to cyItemSave // at creation time, both need to have this if ((plv->ci.style & LVS_OWNERDRAWFIXED) && ListView_IsReportView(plv)) plv->cyItem = plv->cyItemSave; else plv->cyItemSave = plv->cyItem; ListView_OnSetIconSpacing(plv, (LPARAM)-1); ListView_UpdateScrollBars(plv); // sets plv->cItemCol plv->clrBk = CLR_NONE; plv->clrText = CLR_DEFAULT; plv->clrTextBk = CLR_DEFAULT; plv->clrHotlight = CLR_DEFAULT; plv->clrOutline = CLR_DEFAULT; // create the bk brush, and set the imagelists colors if needed ListView_OnSetBkColor(plv, g_clrWindow); // Initialize report view fields plv->xTotalColumnWidth = RECOMPUTE; if (ListView_IsReportView(plv)) ListView_RInitialize(plv, FALSE); if (plv->ci.style & WS_DISABLED) { plv->ci.style &= ~WS_DISABLED; ListView_EnableWindow(plv, FALSE); } // tooltip for unfolding name lables plv->hwndToolTips = CreateWindowEx(WS_EX_TRANSPARENT, TOOLTIPS_CLASS, NULL, WS_POPUP|TTS_NOPREFIX, 0, 0, 0, 0, NULL, NULL, g_hinst, NULL); if (plv->hwndToolTips) { TOOLINFO ti; ti.cbSize = sizeof(ti); ti.uFlags = TTF_TRANSPARENT|TTF_ABSOLUTE; ti.hwnd = plv->ci.hwnd; ti.uId = 0; ti.hinst = NULL; ti.lpszText = LPSTR_TEXTCALLBACK; GetClientRect(plv->ci.hwnd, &ti.rect); SendMessage(plv->hwndToolTips, TTM_ADDTOOL, 0, (LPARAM) &ti); /* Ensure that the tooltips use the same font as the view */ FORWARD_WM_SETFONT(plv->hwndToolTips, plv->hfontLabel, FALSE, SendMessage); } SetTimer(plv->ci.hwnd, IDT_ONECLICKOK, GetDoubleClickTime(), NULL); return TRUE; error1: DPA_Destroy(plv->hdpa); plv->hdpa = NULL; error0: if (plv->plvrangeSel) { plv->plvrangeSel->lpVtbl->Release(plv->plvrangeSel); plv->plvrangeSel = NULL; } if (plv->plvrangeCut) { plv->plvrangeCut->lpVtbl->Release(plv->plvrangeCut); plv->plvrangeCut = NULL; } return FALSE; } void ListView_DeleteHrgnInval(LV* plv) { if (plv->hrgnInval && plv->hrgnInval != (HRGN)ENTIRE_REGION) DeleteObject(plv->hrgnInval); plv->hrgnInval = NULL; } void ListView_OnDestroy(LV* plv) { // // The tooltip window may or may not exist at this point. It // depends if the owning window of the tips is also being destroy. // If so, then the tips are gone already. // if (IsWindow(plv->hwndToolTips)) DestroyWindow(plv->hwndToolTips); if (plv->hCurHot) DestroyCursor(plv->hCurHot); plv->hwndToolTips = NULL; Str_Set(&plv->pszTip, NULL); Str_Set(&plv->pszEmptyText, NULL); TerminateDitherBrush(); if (!ListView_IsOwnerData(plv)) { // Make sure to notify the app ListView_OnDeleteAllItems(plv); } if ((plv->flags & LVF_FONTCREATED) && plv->hfontLabel) { DeleteObject(plv->hfontLabel); // plv->flags &= ~LVF_FONTCREATED; // plv->hwfontLabel = NULL; } if (plv->hfontGroup) { DeleteObject(plv->hfontGroup); } if (plv->hFontHot) { DeleteObject(plv->hFontHot); } if (plv->hbmpWatermark) { DeleteObject(plv->hbmpWatermark); } ListView_DeleteHrgnInval(plv); if (plv->prcWorkAreas) { // This assert is bogus: If the app created work areas then deleted // them, nWorkAreas will be 0 but prcWorkAreas will be non-NULL. // ASSERT(plv->nWorkAreas > 0); LocalFree(plv->prcWorkAreas); } if (plv->hdpaGroups) { DPA_DestroyCallback(plv->hdpaGroups, DestroyGroups, NULL); plv->hdpaGroups = NULL; } } void ListView_OnNCDestroy(LV* plv) { if ((!(plv->ci.style & LVS_SHAREIMAGELISTS)) || ListView_CheckBoxes(plv)) { if (plv->himlState && (plv->himlState != plv->himl) && (plv->himlState != plv->himlSmall)) { ImageList_Destroy(plv->himlState); } } if (!(plv->ci.style & LVS_SHAREIMAGELISTS)) { if (plv->himl) ImageList_Destroy(plv->himl); if (plv->himlSmall) ImageList_Destroy(plv->himlSmall); } if (ListView_IsOwnerData(plv)) { if (plv->plvrangeSel) { plv->plvrangeSel->lpVtbl->Release(plv->plvrangeSel); plv->plvrangeSel = NULL; } if (plv->plvrangeCut) { plv->plvrangeCut->lpVtbl->Release(plv->plvrangeCut); plv->plvrangeCut = NULL; } plv->cTotalItems = 0; } ListView_ReleaseBkImage(plv); if (plv->hbrBk) DeleteBrush(plv->hbrBk); if (plv->hdpa) DPA_Destroy(plv->hdpa); if (plv->hdpaZOrder) DPA_Destroy(plv->hdpaZOrder); ListView_RDestroy(plv); IncrementSearchFree(&plv->is); ListView_SetPtr(plv->ci.hwnd, NULL); if (plv->hTheme) CloseThemeData(plv->hTheme); NearFree(plv); } // sets the background color for the listview // // this creats the brush for drawing the background as well // as sets the imagelists background color if needed BOOL ListView_OnSetBkColor(LV* plv, COLORREF clrBk) { if (plv->clrBk != clrBk) { if (plv->hbrBk) { DeleteBrush(plv->hbrBk); plv->hbrBk = NULL; } if (clrBk != CLR_NONE) { plv->hbrBk = CreateSolidBrush(clrBk); if (!plv->hbrBk) return FALSE; } // don't mess with the imagelist color if things are shared if (!(plv->ci.style & LVS_SHAREIMAGELISTS)) { if (plv->himl) ImageList_SetBkColor(plv->himl, clrBk); if (plv->himlSmall) ImageList_SetBkColor(plv->himlSmall, clrBk); if (plv->himlState) ImageList_SetBkColor(plv->himlState, clrBk); } plv->clrBk = clrBk; } return TRUE; } void InitBrushOrg(LV* plv, HDC hdc) { int x; if (ListView_IsSmallView(plv) || ListView_IsIconView(plv)) { x = plv->ptOrigin.x; } else if (ListView_IsListView(plv)) { x = plv->xOrigin; } else { x = (int)plv->ptlRptOrigin.x; } SetBrushOrgEx(hdc, -x, 0, NULL); } void ListView_InvalidateRegion(LV* plv, HRGN hrgn) { if (hrgn) { if (plv->hrgnInval == NULL) { plv->hrgnInval = hrgn; } else { // union it in if the entire region isn't marked for invalidate if (plv->hrgnInval != (HRGN)ENTIRE_REGION) { UnionRgn(plv->hrgnInval, plv->hrgnInval, hrgn); } DeleteObject(hrgn); } } } // // Used when a watermark is the listview's background (detected via clrTextBk // being CLR_NONE) to perform a flicker-free scroll of the client area, using // an offscreen bitmap // // potential perf issue -- caching DC and/or bitmap instead of create/destroy // on each call // // jeffbog 2/29/96 // void LVSeeThruScroll(LV *plv, LPRECT lprcUpdate) { HDC hdcOff; HBITMAP hbmpOff; int x,y,cx,cy; HDC hdc = GetDC(plv->ci.hwnd); if (!lprcUpdate) { x = y = 0; cx = plv->sizeClient.cx; cy = plv->sizeClient.cy; } else { x = lprcUpdate->left; y = lprcUpdate->top; cx = lprcUpdate->right - x; cy = lprcUpdate->bottom - y; } hdcOff = CreateCompatibleDC(hdc); hbmpOff = CreateCompatibleBitmap(hdc, plv->sizeClient.cx, plv->sizeClient.cy); SelectObject(hdcOff, hbmpOff); SendMessage(plv->ci.hwnd, WM_PRINT, (WPARAM)hdcOff, PRF_CLIENT | PRF_ERASEBKGND); BitBlt(hdc, x, y, cx, cy, hdcOff, x, y, SRCCOPY); ReleaseDC(plv->ci.hwnd, hdc); DeleteDC(hdcOff); DeleteObject(hbmpOff); } void ListView_OnPaint(LV* plv, HDC hdc) { PAINTSTRUCT ps; RECT rcUpdate; HDC hPaintDC = hdc; HDC hMemDC = NULL; HBITMAP hMemBm = NULL; HBITMAP hOldBm; BOOL fInternDC = FALSE; // Before handling WM_PAINT, go ensure everything's recomputed... // if (plv->rcView.left == RECOMPUTE) ListView_Recompute(plv); // If we're in report view, update the header window: it looks // better this way... // if (ListView_IsReportView(plv) && plv->hwndHdr) UpdateWindow(plv->hwndHdr); // If nothing to do (i.e., we recieved a WM_PAINT because // of an RDW_INTERNALPAINT, and we didn't invalidate anything) // don't bother with the Begin/EndPaint. // if (hdc || GetUpdateRect(plv->ci.hwnd, &rcUpdate, FALSE)) { if (!(plv->flags & LVF_VISIBLE)) { plv->flags |= LVF_VISIBLE; // We may try to resize the column ListView_MaybeResizeListColumns(plv, 0, ListView_Count(plv)-1); ListView_UpdateScrollBars(plv); } // this needs to be done before the beginpaint because it clears // out the update region if (!(plv->flags & LVF_REDRAW)) { // add this region to our local invalidate region HRGN hrgn = CreateRectRgn(0, 0, 0,0); if (hrgn) { // ok if GetUpdateRgn fails... then hrgn will still be // and empty region.. GetUpdateRgn(plv->ci.hwnd, hrgn, FALSE); ListView_InvalidateRegion(plv, hrgn); } } // Get device context if (!hdc) { hPaintDC = hdc = BeginPaint(plv->ci.hwnd, &ps); fInternDC = TRUE; } else { GetClipBox(hdc, &ps.rcPaint); } // Skip painting if redrawing is not enabled but complete cycle (EndPaint) if (ListView_RedrawEnabled(plv)) { // Create memory surface and map rendering context if double buffering if (ListView_IsDoubleBuffer(plv)) { // Only make large enough for clipping region hMemDC = CreateCompatibleDC(hdc); if (hMemDC) { hMemBm = CreateCompatibleBitmap(hdc, RECTWIDTH(ps.rcPaint), RECTHEIGHT(ps.rcPaint)); if (hMemBm) { hOldBm = SelectObject(hMemDC, hMemBm); // Offset painting to paint in region OffsetWindowOrgEx(hMemDC, ps.rcPaint.left, ps.rcPaint.top, NULL); } else { DeleteDC(hMemDC); hMemDC = NULL; } } } if (hMemDC) { // Use memory DC if it was created hPaintDC = hMemDC; } if (hPaintDC) { // Setup brush offset for list view scrolling InitBrushOrg(plv, hPaintDC); ListView_DebugDisplayClipRegion(plv, &ps.rcPaint, NULL); // Draw backround in this pass if double buffering, otherwise, it was handled in WM_ERASEBKGND if (ListView_IsDoubleBuffer(plv)) { // Add on buffer offset to scrolling offset POINT ptBrOrg; GetBrushOrgEx(hPaintDC, &ptBrOrg); SetBrushOrgEx(hPaintDC, ptBrOrg.x - ps.rcPaint.left, ptBrOrg.y - ps.rcPaint.top, NULL); ListView_DrawBackground(plv, hPaintDC, &ps.rcPaint); } // Draw foreground ListView_Redraw(plv, hPaintDC, &ps.rcPaint); // Complete double buffering by blitting and freeing off-screen objects if (ListView_IsDoubleBuffer(plv) && hMemDC) { if (plv->flags & LVF_MARQUEE) { HDC h = CreateCompatibleDC(hMemDC); if (h) { HBITMAP hbmp, hbmpOld; BLENDFUNCTION bf = {0}; RECT rcInvalid; RECT rcMarquee = {0, 0, RECTWIDTH(plv->rcMarquee), RECTHEIGHT(plv->rcMarquee)}; IntersectRect(&rcInvalid, &ps.rcPaint, &plv->rcMarquee); if (!IsRectEmpty(&rcInvalid)) { hbmp = CreateCompatibleBitmap(hMemDC, RECTWIDTH(rcInvalid), RECTHEIGHT(rcInvalid)); if (hbmp) { hbmpOld = SelectObject(h, hbmp); FillRectClr(h, &rcMarquee, g_clrMenuHilight); bf.BlendOp = AC_SRC_OVER; bf.SourceConstantAlpha = 70; GdiAlphaBlend(hMemDC, rcInvalid.left, rcInvalid.top, RECTWIDTH(rcInvalid), RECTHEIGHT(rcInvalid), h, 0, 0, RECTWIDTH(rcInvalid), RECTHEIGHT(rcInvalid), bf); SelectObject(h, hbmpOld); DeleteObject(hbmp); } SHOutlineRect(hMemDC, &plv->rcMarquee, g_clrHighlight, g_clrHighlight); } DeleteDC(h); } } BitBlt(hdc, ps.rcPaint.left, ps.rcPaint.top, RECTWIDTH(ps.rcPaint), RECTHEIGHT(ps.rcPaint), hMemDC, ps.rcPaint.left, ps.rcPaint.top, SRCCOPY); SelectObject(hMemDC, hOldBm); DeleteObject(hMemBm); DeleteDC(hMemDC); } } } // Free DC if necessary if (fInternDC) EndPaint(plv->ci.hwnd, &ps); } } void ListView_DrawSimpleBackground(LV *plv, HDC hdc, POINT* ppt, RECT *prcClip) { if (plv->clrBk != CLR_NONE) { // // We just have a simple background color. // FillRect(hdc, prcClip, plv->hbrBk); } else { // // Parent HWND draws the background for us. // POINT pt = {0,0}, ptOrig; MapWindowPoints(plv->ci.hwnd, plv->ci.hwndParent, &pt, 1); //Map it to parent's co-ordinates OffsetWindowOrgEx(hdc, pt.x, pt.y, &ptOrig); SendMessage(plv->ci.hwndParent, WM_ERASEBKGND, (WPARAM)hdc, (LPARAM)0); //Make the parent draw into child's DC SetWindowOrgEx(hdc, ptOrig.x, ptOrig.y, NULL); } } #define SATURATE(x, y) { int ___cTemp; ___cTemp = (x) + ((y + 1) * 100 * (x)) / 1000; if (___cTemp > 0xFF) ___cTemp = 0xFF; (x) = (BYTE)___cTemp; } void SaturateDC(void * pvBitmapBits, int Amount, RECT* prcColumn, RECT* prcImage) { long x, y; long uHeight = RECTHEIGHT(*prcImage); long uWidth = RECTWIDTH(*prcImage); ULONG* pul = (ULONG*)pvBitmapBits; for (y = 0; y < uHeight ;y++) { for (x = 0; x < uWidth; x++) { if (x + prcImage->left >= prcColumn->left && x + prcImage->left <= prcColumn->right) { RGBQUAD* prgb = (RGBQUAD*)&pul[y * uWidth + x]; SATURATE(prgb->rgbRed, Amount); SATURATE(prgb->rgbGreen, Amount); SATURATE(prgb->rgbBlue, Amount); } } } } void SaturateSortColumn(LV* plv, HDC hdc, void * pvBitmapBits, POINT* ppt, RECT* prcClip) { RECT rc; RECT rcUpdate = *prcClip; Header_GetItemRect(plv->hwndHdr, plv->iLastColSort, &rc); OffsetRect(&rc, ppt->x, 0); if (rcUpdate.left < rc.left) rcUpdate.left = rc.left; if (rcUpdate.right > rc.right) rcUpdate.right = rc.right; if (rcUpdate.left < rcUpdate.right || IntersectRect(&rc, &rcUpdate, prcClip)) { SaturateDC(pvBitmapBits, 0, &rcUpdate, prcClip); } } HDC PrepBackgroundDIBSection(HDC hdcDest, RECT* prc, void ** ppvBitmap, HBITMAP* phbmpOld) { HDC hdcRet = CreateCompatibleDC(hdcDest); if (hdcRet) { HBITMAP hbmp; BITMAPINFO bi = {0}; bi.bmiHeader.biSize = sizeof(bi.bmiHeader); bi.bmiHeader.biWidth = RECTWIDTH(*prc); bi.bmiHeader.biHeight = RECTHEIGHT(*prc); bi.bmiHeader.biPlanes = 1; bi.bmiHeader.biBitCount = 32; bi.bmiHeader.biCompression = BI_RGB; hbmp = CreateDIBSection(hdcRet, &bi, DIB_RGB_COLORS, ppvBitmap, NULL, 0); *phbmpOld = (HBITMAP)SelectObject(hdcRet, hbmp); SetViewportOrgEx(hdcRet, -prc->left, -prc->top, NULL); } return hdcRet; } void CleanupBackgroundDIBSection(HDC hdc, HBITMAP hbmpOld) { if (hdc) { HBITMAP hbmp = SelectObject(hdc, hbmpOld); if (hbmp) DeleteObject(hbmp); DeleteDC(hdc); } } void ListView_DrawBackground(LV *plv, HDC hdc, RECT *prcClip) { HRGN hrgnClipSave; RECT rcClip; POINT ptBackOrg = {0}; // // Compute ptBackOrg (aka scrolling offset), based on view style. // switch (plv->wView) { case LV_VIEW_LIST: ptBackOrg.x = -plv->xOrigin; ptBackOrg.y = 0; break; case LV_VIEW_DETAILS: ptBackOrg.x = -plv->ptlRptOrigin.x; ptBackOrg.y = -plv->ptlRptOrigin.y + plv->yTop; break; default: ptBackOrg.x = -plv->ptOrigin.x; ptBackOrg.y = -plv->ptOrigin.y; break; } // Optimize the common/simple case if (!(plv->pImgCtx && plv->fImgCtxComplete)) { ListView_DrawSimpleBackground(plv, hdc, &ptBackOrg, prcClip); if (ListView_IsWatermarked(plv)) { HDC hdcMem = CreateCompatibleDC(hdc); if (hdcMem) { HBITMAP hbmp = (HBITMAP)SelectObject(hdcMem, plv->hbmpWatermark); RECT rcWatermark; GetClientRect(plv->ci.hwnd, &rcWatermark); rcWatermark.left = rcWatermark.right - plv->szWatermark.cx; rcWatermark.top = rcWatermark.bottom - plv->szWatermark.cy; BitBlt(hdc, rcWatermark.left, rcWatermark.top, plv->szWatermark.cx, plv->szWatermark.cy, hdcMem, 0, 0, SRCCOPY); SelectObject(hdcMem, hbmp); DeleteDC(hdcMem); } } if (plv->wView == LV_VIEW_DETAILS && plv->iLastColSort != -1 && !plv->fGroupView) { RECT rcUpdate = *prcClip; RECT rc; COLORREF cr; Header_GetItemRect(plv->hwndHdr, plv->iLastColSort, &rc); OffsetRect(&rc, ptBackOrg.x, 0); if (rcUpdate.left < rc.left) rcUpdate.left = rc.left; if (rcUpdate.right > rc.right) rcUpdate.right = rc.right; cr = GetSortColor(10, plv->clrBk); FillRectClr(hdc, &rcUpdate, cr); } return; } // // Save the old clipping region, // since we whack on it a lot. // hrgnClipSave = CreateRectRgnIndirect(prcClip); if (hrgnClipSave) { if (GetClipRgn(hdc, hrgnClipSave) <= 0) { DeleteObject(hrgnClipSave); hrgnClipSave = NULL; } } // // Clip the clipping region to the caller's rectangle, // and save the final clipping rectangle in rcClip. // if (prcClip != NULL) { IntersectClipRect(hdc, prcClip->left, prcClip->top, prcClip->right, prcClip->bottom); } GetClipBox(hdc, &rcClip); if (plv->pImgCtx && plv->fImgCtxComplete) { RECT rcImage, rcClient; ULONG ulState; SIZE sizeImg; ListView_Realize(plv, hdc, TRUE, FALSE); switch (plv->ulBkImageFlags & LVBKIF_STYLE_MASK) { case LVBKIF_STYLE_TILE: { HDC hdcBackBuffer = hdc; HBITMAP hbmpOld; void * pvBits = NULL; POINT ptBackTile = {0}; if (plv->wView == LV_VIEW_DETAILS && plv->iLastColSort != -1) { hdcBackBuffer = PrepBackgroundDIBSection(hdc, prcClip, &pvBits, &hbmpOld); if (hdcBackBuffer == NULL) hdcBackBuffer = hdc; } if (!plv->fListviewWatermarkBackgroundImages) ptBackTile = ptBackOrg; if (plv->ulBkImageFlags & LVBKIF_FLAG_TILEOFFSET) { // These offsets are in pixels, not percent (sorry) ptBackTile.x -= plv->xOffsetPercent; ptBackTile.y -= plv->yOffsetPercent; } IImgCtx_Tile(plv->pImgCtx, hdcBackBuffer, &ptBackTile, prcClip, NULL); if (hdcBackBuffer != hdc) { SaturateSortColumn(plv, hdcBackBuffer, pvBits, &ptBackOrg, prcClip); BitBlt(hdc, prcClip->left, prcClip->top, RECTWIDTH(*prcClip), RECTHEIGHT(*prcClip), hdcBackBuffer, prcClip->left, prcClip->top, SRCCOPY); CleanupBackgroundDIBSection(hdcBackBuffer, hbmpOld); } } ExcludeClipRect(hdc, prcClip->left, prcClip->top, prcClip->right, prcClip->bottom); break; case LVBKIF_STYLE_NORMAL: // // Start with the base image. // IImgCtx_GetStateInfo(plv->pImgCtx, &ulState, &sizeImg, FALSE); rcImage.left = 0; rcImage.top = 0; rcImage.right = sizeImg.cx; rcImage.bottom = sizeImg.cy; // // Adjust for caller offsets. // GetClientRect(plv->ci.hwnd, &rcClient); if (plv->xOffsetPercent) { LONG dx = plv->xOffsetPercent * (rcClient.right - sizeImg.cx) / 100; rcImage.left += dx; rcImage.right += dx; } if (plv->yOffsetPercent) { LONG dy = plv->yOffsetPercent * (rcClient.bottom - sizeImg.cy) / 100; rcImage.top += dy; rcImage.bottom += dy; } // // Adjust for ptBackOrg (scrolling offset). // rcImage.left += ptBackOrg.x; rcImage.top += ptBackOrg.y; rcImage.right += ptBackOrg.x; rcImage.bottom += ptBackOrg.y; // // Draw the image, if necessary. // if (RectVisible(hdc, &rcImage)) { IImgCtx_Draw(plv->pImgCtx, hdc, &rcImage); ExcludeClipRect(hdc, rcImage.left, rcImage.top, rcImage.right, rcImage.bottom); } break; } } // // Now draw the rest of the background. // if (RectVisible(hdc, prcClip)) { ListView_DrawSimpleBackground(plv, hdc, &ptBackOrg, prcClip); } // // Restore old clipping region. // SelectClipRgn(hdc, hrgnClipSave); if (hrgnClipSave) { DeleteObject(hrgnClipSave); } } BOOL ListView_OnEraseBkgnd(LV *plv, HDC hdc) { // If redraw is turned off, still process erase bk if (ListView_IsDoubleBuffer(plv) && (plv->flags & LVF_REDRAW)) { // No erase, will happen in WM_PAINT handler (ListView_OnPaint) return FALSE; } else { RECT rcClip; // // We draw our own background, erase with it. // GetClipBox(hdc, &rcClip); ListView_DrawBackground(plv, hdc, &rcClip); return TRUE; } } void ListView_OnCommand(LV* plv, int id, HWND hwndCtl, UINT codeNotify) { if (hwndCtl == plv->hwndEdit) { switch (codeNotify) { case EN_UPDATE: // We don't want flicker during replacing current selection // as we use selection for IME composition. // if ((g_fDBCSInputEnabled) && (plv->flags & LVF_INSERTINGCOMP)) break; // We will use the ID of the window as a Dirty flag... if (IsWindowVisible(plv->hwndEdit)) { SetWindowID(plv->hwndEdit, 1); ListView_SetEditSize(plv); } break; case EN_KILLFOCUS: // We lost focus, so dismiss edit and save changes // (Note that the owner might reject the change and restart // edit mode, which traps the user. Owners need to give the // user a way to get out.) // // // Fix horrible undocumented hanging problem: LVN_ENDLABELEDIT // is sent in response to EN_KILLFOCUS, which is send in response // to WM_KILLFOCUS, and it is undocumented that you cannot display // UI during WM_KILLFOCUS when a journal record hook is active, // because the presence of a hook forces serialization of activation, // and so when you put up UI, you generate activation changes, which // get stuck because you haven't finished responding to the previous // WM_KILLFOCUS message yet. // // See NT bug 414634. // if (InSendMessage()) ReplyMessage(0); if (!ListView_DismissEdit(plv, FALSE)) return; break; case HN_BEGINDIALOG: // pen windows is bringing up a dialog ASSERT(GetSystemMetrics(SM_PENWINDOWS)); // only on a pen system plv->fNoDismissEdit = TRUE; break; case HN_ENDDIALOG: // pen windows has destroyed dialog ASSERT(GetSystemMetrics(SM_PENWINDOWS)); // only on a pen system plv->fNoDismissEdit = FALSE; break; } // Forward edit control notifications up to parent // if (IsWindow(hwndCtl)) FORWARD_WM_COMMAND(plv->ci.hwndParent, id, hwndCtl, codeNotify, SendMessage); } } void ListView_OnWindowPosChanged(LV* plv, const WINDOWPOS* lpwpos) { if (!lpwpos || !(lpwpos->flags & SWP_NOSIZE)) { RECT rc; int iOldSlots; // Update scrollbars first, since ListView_OnEnsureVisible requires accurate scroll info ListView_UpdateScrollBars(plv); if (ListView_IsOwnerData(plv) && ListView_IsSlotView(plv)) { iOldSlots = ListView_GetSlotCount(plv, TRUE, NULL, NULL); } GetClientRect(plv->ci.hwnd, &rc); plv->sizeClient.cx = rc.right; plv->sizeClient.cy = rc.bottom; if (ListView_IsAutoArrangeView(plv)) { // Call off to the arrange function. ListView_ArrangeOrSnapToGrid(plv); } if (ListView_IsOwnerData(plv)) { plv->rcView.left = RECOMPUTE; ListView_Recompute(plv); ListView_DismissEdit(plv, FALSE); if (ListView_IsSlotView(plv)) { // Uses the int iNewSlots = ListView_GetSlotCount(plv, TRUE, NULL, NULL); if ((iNewSlots != iOldSlots) && (ListView_Count(plv) > min(iNewSlots, iOldSlots))) RedrawWindow(plv->ci.hwnd, NULL, NULL, RDW_INVALIDATE | RDW_ERASE); } } ListView_RInitialize(plv, TRUE); if (ListView_IsWatermarked(plv)) { GetClientRect(plv->ci.hwnd, &rc); rc.left = rc.right - plv->szWatermark.cx; rc.top = rc.bottom - plv->szWatermark.cy; InvalidateRect(plv->ci.hwnd, &rc, TRUE); } } } void ListView_InvalidateSelectedOrCutOwnerData(LV* plv, ILVRange *plvrangeSel) { UINT rdwFlags = RDW_INVALIDATE; int cItem = ListView_Count(plv); DWORD dwType = plv->wView; int i; RECT rcView; ASSERT(ListView_IsOwnerData(plv)); ASSERT(plv); GetClientRect(plv->ci.hwnd, &rcView); if (plv->clrTextBk == CLR_NONE || (plv->himl && (plv->clrBk != ImageList_GetBkColor(plv->himl)))) { // always do an erase, otherwise the text background won't paint right rdwFlags |= RDW_ERASE; } // calculate start of items and end of items visible on the view // switch (dwType) { case LV_VIEW_DETAILS: i = ListView_RYHitTest(plv, rcView.top); cItem = ListView_RYHitTest(plv, rcView.bottom) + 1; break; case LV_VIEW_LIST: i = ListView_LCalcViewItem(plv, rcView.left, rcView.top); cItem = ListView_LCalcViewItem(plv, rcView.right, rcView.bottom) + 1; break; default: ListView_CalcMinMaxIndex(plv, &rcView, &i, &cItem); break; } i = max(i, 0); cItem = min(ListView_Count(plv), cItem); if (cItem > i) { ListView_NotifyCacheHint(plv, i, cItem-1); } for (; i < cItem; i++) { if (plvrangeSel->lpVtbl->IsSelected(plvrangeSel, i) == S_OK) { ListView_InvalidateItem(plv, i, FALSE, rdwFlags); } } } void ListView_RedrawSelection(LV* plv) { if (ListView_IsOwnerData(plv)) { ListView_InvalidateSelectedOrCutOwnerData(plv, plv->plvrangeSel); } else { int i = -1; while ((i = ListView_OnGetNextItem(plv, i, LVNI_SELECTED)) != -1) { ListView_InvalidateItem(plv, i, TRUE, RDW_INVALIDATE | RDW_ERASE); } if (ListView_IsReportView(plv)) { int iEnd = ListView_RYHitTest(plv, plv->sizeClient.cy) + 1; iEnd = min(iEnd, ListView_Count(plv)); // if we're in report mode, sub items may have selection focus for (i = ListView_RYHitTest(plv, 0); i < iEnd; i++) { int iCol; for (iCol = 1; iCol < plv->cCol; iCol++) { LISTSUBITEM lsi; ListView_GetSubItem(plv, i, iCol, &lsi); if (lsi.state & LVIS_SELECTED) { ListView_InvalidateItem(plv, i, FALSE, RDW_INVALIDATE | RDW_ERASE); } break; } } } } UpdateWindow(plv->ci.hwnd); } void ListView_OnSetFocus(LV* plv, HWND hwndOldFocus) { ASSERT(gcWheelDelta == 0); // due to the way listview call SetFocus on themselves on buttondown, // the window can get a strange sequence of focus messages: first // set, then kill, and then set again. since these are not really // focus changes, ignore them and only handle "real" cases. // // But still send out the accessibility notification because USER // has already pushed focus back to the listview instead of to the // focus item. if (hwndOldFocus == plv->ci.hwnd) { ListView_NotifyFocusEvent(plv); return; } plv->flags |= ListView_HideLabels(plv) ? LVF_FOCUSED : LVF_FOCUSED | LVF_UNFOLDED; if (IsWindowVisible(plv->ci.hwnd)) { if (plv->iFocus != -1) { ListView_InvalidateItem(plv, plv->iFocus, TRUE, RDW_INVALIDATE | RDW_ERASE); ListView_NotifyFocusEvent(plv); } ListView_RedrawSelection(plv); } // Let the parent window know that we are getting the focus. CCSendNotify(&plv->ci, NM_SETFOCUS, NULL); } void ListView_OnKillFocus(LV* plv, HWND hwndNewFocus) { // Reset wheel scroll amount gcWheelDelta = 0; // due to the way listview call SetFocus on themselves on buttondown, // the window can get a strange sequence of focus messages: first // set, then kill, and then set again. since these are not really // focus changes, ignore them and only handle "real" cases. if (!plv || hwndNewFocus == plv->ci.hwnd) return; ListView_CancelTipTrack(plv); plv->flags &= ~(LVF_FOCUSED|LVF_UNFOLDED); // Blow this off if we are not currently visible (being destroyed!) if (IsWindowVisible(plv->ci.hwnd)) { if (plv->iFocus != -1) { UINT fRedraw = RDW_INVALIDATE; if (plv->clrTextBk == CLR_NONE || plv->fListviewShadowText) fRedraw |= RDW_ERASE; ListView_InvalidateFoldedItem(plv, plv->iFocus, TRUE, fRedraw); } ListView_RedrawSelection(plv); } // Let the parent window know that we are losing the focus. CCSendNotify(&plv->ci, NM_KILLFOCUS, NULL); IncrementSearchString(&plv->is, 0, NULL); } void ListView_DeselectAll(LV* plv, int iDontDeselect) { int i = -1; int nSkipped = 0; BOOL fWasSelected = FALSE; if (iDontDeselect != -1) { if (ListView_OnGetItemState(plv, iDontDeselect, LVIS_SELECTED)) fWasSelected = TRUE; } if (ListView_IsOwnerData(plv)) { // if there's only one item selected, and that item is the iDontDeselect // then our work is done... plv->plvrangeSel->lpVtbl->CountIncluded(plv->plvrangeSel, &plv->nSelected); if (plv->nSelected == 1 && fWasSelected) return; ListView_InvalidateSelectedOrCutOwnerData(plv, plv->plvrangeSel); ListView_OnSetItemState(plv, -1, 0, LVIS_SELECTED); if (fWasSelected) { ListView_OnSetItemState(plv, iDontDeselect, LVIS_SELECTED, LVIS_SELECTED); nSkipped = 1; } } else { if (iDontDeselect != plv->iFocus) { ListView_OnSetItemState(plv, plv->iFocus, 0, LVIS_SELECTED); } while ((plv->nSelected - nSkipped) && (i = ListView_OnGetNextItem(plv, i, LVNI_SELECTED)) != -1) { if (i != iDontDeselect) { ListView_OnSetItemState(plv, i, 0, LVIS_SELECTED); } else { if (fWasSelected) { nSkipped++; } } } } RIPMSG((plv->nSelected - nSkipped) == 0, "ListView_DeselectAll: Do not refuse a deselect when telling listview to Deselect all."); plv->nSelected = nSkipped; } // toggle the selection state of an item void ListView_ToggleSelection(LV* plv, int iItem) { UINT cur_state; if (iItem != -1) { cur_state = ListView_OnGetItemState(plv, iItem, LVIS_SELECTED); ListView_OnSetItemState(plv, iItem, cur_state ^ LVIS_SELECTED, LVIS_SELECTED); } } // Selects (or toggles) a range of items in the list. // The curent iFocus is the starting location // iItem - is the ending item // fToggle - Well set all of the selection state of all of the items to // inverse the starting location // void ListView_SelectRangeTo(LV* plv, int iItem, BOOL fResetRest) { int iMin, iMax; int i = -1; UINT uSelVal = LVIS_SELECTED; if (plv->iMark == -1) { ListView_SetFocusSel(plv, iItem, TRUE, TRUE, FALSE); return; } if (!fResetRest) uSelVal = ListView_OnGetItemState(plv, plv->iMark, LVIS_SELECTED); // If we are in report view or list view we simply walk through the // indexes to see which items to select or deselect. otherwise it // is is based off of the location of the objects being within the // rectangle that is defined by if (ListView_IsListView(plv) || (ListView_IsReportView(plv) && !plv->fGroupView)) { iMin = min(iItem, plv->iMark); iMax = max(iItem, plv->iMark); if (ListView_IsOwnerData(plv)) { if (fResetRest) { ListView_DeselectAll(plv, -1); } if (iMax > iMin) { if (LVIS_SELECTED & uSelVal) { if (FAILED(plv->plvrangeSel->lpVtbl->IncludeRange(plv->plvrangeSel, iMin, iMax))) return; } else { if (FAILED(plv->plvrangeSel->lpVtbl->ExcludeRange(plv->plvrangeSel, iMin, iMax))) return; } ListView_SendODChangeAndInvalidate(plv, iMin, iMax, uSelVal ^ LVIS_SELECTED, uSelVal); } else { ListView_OnSetItemState(plv, iMin, uSelVal, LVIS_SELECTED); } } else { if (fResetRest) { while ((i = ListView_OnGetNextItem(plv, i, LVNI_SELECTED)) != -1) { if (i < iMin || i > iMax) ListView_OnSetItemState(plv, i, 0, LVIS_SELECTED); } } while (iMin <= iMax) { ListView_OnSetItemState(plv, iMin, uSelVal, LVIS_SELECTED); iMin++; } } } else { RECT rcTemp; RECT rcTemp2; RECT rcBounding; int iFirstItem = (plv->iMark < iItem)? plv->iMark: iItem; int iSecondItem = (plv->iMark > iItem)? plv->iMark: iItem; ListView_GetRects(plv, iFirstItem, QUERY_DEFAULT, NULL, NULL, NULL, &rcTemp); ListView_GetRects(plv, iSecondItem, QUERY_DEFAULT, NULL, NULL, NULL, &rcTemp2); UnionRect(&rcBounding, &rcTemp, &rcTemp2); ListView_CalcMinMaxIndex(plv, &rcBounding, &iMin, &iMax); if (ListView_IsOwnerData(plv) && (iMax > iMin)) { if (fResetRest) { ListView_DeselectAll(plv, -1); } iMax = min(iMax, ListView_Count(plv)); iMin = max(iMin, 0); if (LVIS_SELECTED & uSelVal) { if (FAILED(plv->plvrangeSel->lpVtbl->IncludeRange(plv->plvrangeSel, iMin, iMax - 1))) return; } else { if (FAILED(plv->plvrangeSel->lpVtbl->ExcludeRange(plv->plvrangeSel, iMin, iMax - 1))) return; } ListView_SendODChangeAndInvalidate(plv, iMin, iMax, uSelVal ^ LVIS_SELECTED, uSelVal); } else { int iZ; POINT pt; RECT rcItem; for (i = 0; i < ListView_Count(plv); i++) { ListView_GetRects(plv, i, QUERY_DEFAULT, NULL, NULL, NULL, &rcItem); pt.x = (rcItem.right + rcItem.left) / 2; // center of item pt.y = (rcItem.bottom + rcItem.top) / 2; // Is the item within the y bound of the first and last item? if (pt.y > rcTemp.top && pt.y < rcTemp2.bottom) { // Yes. Check to see if the item is in the first row. if (pt.y < rcTemp.bottom) { // It is. Then check to see if it's before the first item in that row. if (pt.x < rcTemp.left) { // It is. Then this item is not to be selected. if (fResetRest) ListView_OnSetItemState(plv, i, 0, LVIS_SELECTED); // Continue to the next item continue; } } // Is the item in the last row? if (pt.y > rcTemp2.top) { // Yes. Is it after the last item in the selection? if (pt.x > rcTemp2.right) { // It is. Then this item is not to be selected. if (fResetRest) ListView_OnSetItemState(plv, i, 0, LVIS_SELECTED); // Continue to the next item continue; } } // The item is in the selection range. Go ahead and select it if (!ListView_IsOwnerData(plv)) { iZ = ListView_ZOrderIndex(plv, i); if (iZ > 0) DPA_InsertPtr(plv->hdpaZOrder, 0, DPA_DeletePtr(plv->hdpaZOrder, iZ)); } ListView_OnSetItemState(plv, i, uSelVal, LVIS_SELECTED); } else if (fResetRest) ListView_OnSetItemState(plv, i, 0, LVIS_SELECTED); } } } } // makes an item the focused item and optionally selects it // // in: // iItem item to get the focus // fSelectAlso select this item as well as set it as the focus // fDeselectAll deselect all items first // fToggleSel toggle the selection state of the item // // returns: // index of focus item (if focus change was refused) int ListView_SetFocusSel(LV* plv, int iItem, BOOL fSelectAlso, BOOL fDeselectAll, BOOL fToggleSel) { int iFocus = plv->iFocus; // if we're single sel mode, don't bother with this because // the set item will do it for us if (!(plv->ci.style & LVS_SINGLESEL) && (fDeselectAll)) ListView_DeselectAll(plv, -1); if (iItem != plv->iFocus) { // remove the old focus if (plv->iFocus != -1) { // If he refuses to give up the focus, bail out. if (!ListView_OnSetItemState(plv, plv->iFocus, 0, LVIS_FOCUSED)) return plv->iFocus; } } if (!ListView_IsOwnerData(plv)) { if (fSelectAlso) { if (ListView_IsIconView(plv) || ListView_IsSmallView(plv) || ListView_IsTileView(plv)) { int iZ = ListView_ZOrderIndex(plv, iItem); if (iZ > 0) DPA_InsertPtr(plv->hdpaZOrder, 0, DPA_DeletePtr(plv->hdpaZOrder, iZ)); } } } /* Ensure that when moving focus that we refresh the previous focus owner properly. */ if (iFocus != -1 && iFocus != plv->iFocus && (plv->flags & LVF_UNFOLDED)) ListView_InvalidateFoldedItem(plv, iFocus, FALSE, RDW_INVALIDATE); if (plv->iMark == -1) plv->iMark = iItem; SetTimer(plv->ci.hwnd, IDT_SCROLLWAIT, GetDoubleClickTime(), NULL); plv->flags |= LVF_SCROLLWAIT; if (fToggleSel) { ListView_ToggleSelection(plv, iItem); ListView_OnSetItemState(plv, iItem, LVIS_FOCUSED, LVIS_FOCUSED); } else { UINT flags = ((fSelectAlso || plv->ci.style & LVS_SINGLESEL) ? (LVIS_SELECTED | LVIS_FOCUSED) : LVIS_FOCUSED); ListView_OnSetItemState(plv, iItem, flags, flags); } return iItem; } UINT GetLVKeyFlags() { UINT uFlags = 0; if (GetKeyState(VK_MENU) < 0) uFlags |= LVKF_ALT; if (GetKeyState(VK_CONTROL) < 0) uFlags |= LVKF_CONTROL; if (GetKeyState(VK_SHIFT) < 0) uFlags |= LVKF_SHIFT; return uFlags; } void ListView_OnKey(LV* plv, UINT vk, BOOL fDown, int cRepeat, UINT flags) { UINT lvni = 0; int iNewFocus; BOOL fCtlDown; BOOL fShiftDown; LV_KEYDOWN nm; HWND hwnd = plv->ci.hwnd; if (!fDown) return; // Cancel manual tip track if any key is pressed ListView_CancelTipTrack(plv); // Swap the left and right arrow key if the control is mirrored. vk = RTLSwapLeftRightArrows(&plv->ci, vk); //prevent any change in selected items before the dbl click timer goes off //so that we don't launch wrong item(s) if (plv->exStyle & LVS_EX_ONECLICKACTIVATE && plv->fOneClickHappened && plv->fOneClickOK) { //if a key is pressed with a mouse click with one click activate and double click //timer, we end up setting up a timer and then processing the keydown //this causes an item to be launched right away (from this code) and in case //of return being pressed it causes double activation //prevent these cases: if (vk == VK_SHIFT || vk == VK_CONTROL || vk == VK_MENU || vk == VK_RETURN) return; KillTimer(plv->ci.hwnd, IDT_ONECLICKHAPPENED); plv->fOneClickHappened = FALSE; CCSendNotify(&plv->ci, LVN_ITEMACTIVATE, &(plv->nmOneClickHappened.hdr)); if (!IsWindow(hwnd)) return; } // Notify nm.wVKey = (WORD) vk; nm.flags = flags; if (CCSendNotify(&plv->ci, LVN_KEYDOWN, &nm.hdr)) { plv->iPuntChar++; return; } else if (plv->iPuntChar) { // this is tricky... if we want to punt the char, just increment the // count. if we do NOT, then we must clear the queue of WM_CHAR's // this is to preserve the iPuntChar to mean "punt the next n WM_CHAR messages MSG msg; while(plv->iPuntChar && PeekMessage(&msg, plv->ci.hwnd, WM_CHAR, WM_CHAR, PM_REMOVE)) { plv->iPuntChar--; } ASSERT(!plv->iPuntChar); } if (ListView_Count(plv) == 0) // don't blow up on empty list return; fCtlDown = GetKeyState(VK_CONTROL) < 0; fShiftDown = GetKeyState(VK_SHIFT) < 0; switch (vk) { case VK_SPACE: // If shift (extend) or control (disjoint) select, // then toggle selection state of focused item. if (fCtlDown) { plv->iMark = plv->iFocus; ListView_ToggleSelection(plv, plv->iFocus); plv->iPuntChar++; } if (fShiftDown) { ListView_SelectRangeTo(plv, plv->iFocus, TRUE); } if (ListView_CheckBoxes(plv)) { if (plv->iFocus != -1) ListView_HandleStateIconClick(plv, plv->iFocus); if (ListView_IsSimpleSelect(plv)) { int iToggle = -1; while ((iToggle = ListView_OnGetNextItem(plv, iToggle, LVNI_SELECTED)) != -1) { if (plv->iFocus != iToggle) { ListView_HandleStateIconClick(plv, iToggle); } } } } //notify of navigation key usage CCNotifyNavigationKeyUsage(&(plv->ci), UISF_HIDEFOCUS); return; case VK_RETURN: CCSendNotify(&plv->ci, NM_RETURN, NULL); /// some (comdlg32 for example) destroy on double click // we need to bail if that happens because plv is no longer valid if (!IsWindow(hwnd)) return; { NMITEMACTIVATE nm; nm.iItem = plv->iFocus; nm.iSubItem = 0; nm.uChanged = 0; nm.ptAction.x = -1; nm.ptAction.y = -1; nm.uKeyFlags = GetLVKeyFlags(); CCSendNotify(&plv->ci, LVN_ITEMACTIVATE, &nm.hdr); if (!IsWindow(hwnd)) return; } //notify of navigation key usage CCNotifyNavigationKeyUsage(&(plv->ci), UISF_HIDEFOCUS); return; case VK_ADD: if (ListView_IsReportView(plv) && (GetKeyState(VK_CONTROL) < 0)) { HCURSOR hcurPrev; int i; hcurPrev = SetCursor(LoadCursor(NULL, IDC_WAIT)); for (i=0; i < plv->cCol; i++) { ListView_RSetColumnWidth(plv, i, -1); } SetCursor(hcurPrev); //notify of navigation key usage CCNotifyNavigationKeyUsage(&(plv->ci), UISF_HIDEFOCUS); return; } } if (GetKeyState(VK_MENU) < 0) return; // For a single selection listview, disable extending the selection // by turning off the keyboard modifiers. if (plv->ci.style & LVS_SINGLESEL) { fCtlDown = FALSE; fShiftDown = FALSE; } // // Let the Arrow function attempt to process the key. // iNewFocus = ListView_Arrow(plv, plv->iFocus, vk); // If control (disjoint) selection, don't change selection. // If shift (extend) or control selection, don't deselect all. // if (iNewFocus != -1) { if (fShiftDown) { ListView_SelectRangeTo(plv, iNewFocus, TRUE); ListView_SetFocusSel(plv, iNewFocus, FALSE, FALSE, FALSE); } else { if (!fCtlDown) plv->iMark = iNewFocus; ListView_SetFocusSel(plv, iNewFocus, !fCtlDown, !fShiftDown && !fCtlDown, FALSE); } IncrementSearchString(&plv->is, 0, NULL); CCPlaySound(c_szSelect); ListView_OnKeyboardSelected(plv, iNewFocus); } // on keyboard movement, scroll immediately. if (ListView_CancelScrollWait(plv)) { ListView_OnEnsureVisible(plv, plv->iFocus, FALSE); UpdateWindow(plv->ci.hwnd); } //notify of navigation key usage CCNotifyNavigationKeyUsage(&(plv->ci), UISF_HIDEFOCUS); } // // LVN_INCREMENTALSEARCH gives the app the opportunity to customize // incremental search. For example, if the items are numeric, // the app can do numerical search instead of string search. // // App sets pnmfi->lvfi.lParam to the result of the incremental search, // or to -2 to fai the search and just beep. // // App can return 2 to indicate that all processing should stop, if // app wants to take over incremental search completely. // BOOL ListView_IncrementalSearch(LV *plv, int iStartFrom, LPNMLVFINDITEM pnmfi, int *pi) { INT_PTR fRc; ASSERT(!(pnmfi->lvfi.flags & LVFI_PARAM)); pnmfi->lvfi.lParam = -1; fRc = CCSendNotify(&plv->ci, LVN_INCREMENTALSEARCH, &pnmfi->hdr); *pi = (int)pnmfi->lvfi.lParam; // Cannot just return fRc because some apps return 1 to all WM_NOTIFY's return fRc == 2; } // Now only Korean version is interested in incremental search with composition string. LPTSTR GET_COMP_STRING(HIMC hImc, DWORD dwFlags) { LONG iNumComp; PTSTR pszCompStr; iNumComp = ImmGetCompositionString(hImc, dwFlags, NULL, 0); pszCompStr = (PTSTR)LocalAlloc(LPTR, sizeof(TCHAR)*(iNumComp+1)); if (pszCompStr) { if (iNumComp) ImmGetCompositionString(hImc, dwFlags, pszCompStr, iNumComp+1); pszCompStr[iNumComp] = TEXT('\0'); } return pszCompStr; } #define FREE_COMP_STRING(pszCompStr) LocalFree((HLOCAL)(pszCompStr)) BOOL ListView_OnImeComposition(LV* plv, WPARAM wParam, LPARAM lParam) { LPTSTR lpsz; NMLVFINDITEM nmfi; int i; int iStartFrom = -1; int iLen; int iCount; HIMC hImc; TCHAR *pszCompStr; BOOL fRet = TRUE; iCount = ListView_Count(plv); if (!iCount || plv->iFocus == -1) return fRet; if (hImc = ImmGetContext(plv->ci.hwnd)) { if (lParam & GCS_RESULTSTR) { fRet = FALSE; pszCompStr = GET_COMP_STRING(hImc, GCS_RESULTSTR); if (pszCompStr) { IncrementSearchImeCompStr(&plv->is, FALSE, pszCompStr, &lpsz); FREE_COMP_STRING(pszCompStr); } } if (lParam & GCS_COMPSTR) { fRet = TRUE; pszCompStr = GET_COMP_STRING(hImc, GCS_COMPSTR); if (pszCompStr) { if (IncrementSearchImeCompStr(&plv->is, TRUE, pszCompStr, &lpsz)) iStartFrom = plv->iFocus; else iStartFrom = ((plv->iFocus - 1) + iCount)% iCount; nmfi.lvfi.flags = LVFI_SUBSTRING | LVFI_STRING | LVFI_WRAP; nmfi.lvfi.psz = lpsz; iLen = lstrlen(lpsz); // special case space as the first character if ((iLen == 1) && (*lpsz == TEXT(' '))) { if (plv->iFocus != -1) { ListView_OnSetItemState(plv, plv->iFocus, LVIS_SELECTED, LVIS_SELECTED); IncrementSearchString(&plv->is, 0, NULL); } //notify of navigation key usage CCNotifyNavigationKeyUsage(&(plv->ci), UISF_HIDEFOCUS); return fRet; } // Give caller full string in case they want to do something custom if (ListView_IncrementalSearch(plv, iStartFrom, &nmfi, &i)) return fRet; if (iLen > 0 && SameChars(lpsz, lpsz[0])) { // The user has been typing the same char over and over again. // Switch from incremental search to Windows 3.1 style search. iStartFrom = plv->iFocus; nmfi.lvfi.psz = lpsz + iLen - 1; } if (i == -1) i = ListView_OnFindItem(plv, iStartFrom, &nmfi.lvfi); if (!ListView_IsValidItemNumber(plv, i)) { i = -1; } TraceMsg(TF_LISTVIEW, "CIme listsearch %d %s %d", (LPTSTR)lpsz, (LPTSTR)lpsz, i); if (i != -1) { ListView_SetFocusSel(plv, i, TRUE, TRUE, FALSE); plv->iMark = i; if (ListView_CancelScrollWait(plv)) ListView_OnEnsureVisible(plv, i, FALSE); } else { // Don't beep on spaces, we use it for selection. IncrementSearchBeep(&plv->is); } //notify of navigation key usage CCNotifyNavigationKeyUsage(&(plv->ci), UISF_HIDEFOCUS); FREE_COMP_STRING(pszCompStr); } } ImmReleaseContext(plv->ci.hwnd, hImc); } return fRet; } // REVIEW: We will want to reset ichCharBuf to 0 on certain conditions, // such as: focus change, ENTER, arrow key, mouse click, etc. // void ListView_OnChar(LV* plv, UINT ch, int cRepeat) { LPTSTR lpsz; NMLVFINDITEM nmfi; int i; int iStartFrom = -1; int iLen; int iCount; iCount = ListView_Count(plv); if (!iCount) return; // Don't search for chars that cannot be in a file name (like ENTER and TAB) // The Polish keyboard layout uses CTRL+ALT to // enter some normal letters, so don't punt if the CTRL key is down or // people in Poland are in trouble! We need to fix this. NTRAID 5262. if (ch < TEXT(' '))// || GetKeyState(VK_CONTROL) < 0) { IncrementSearchString(&plv->is, 0, NULL); return; } if (IncrementSearchString(&plv->is, ch, &lpsz)) iStartFrom = plv->iFocus; else iStartFrom = ((plv->iFocus - 1) + iCount)% iCount; nmfi.lvfi.flags = LVFI_SUBSTRING | LVFI_STRING | LVFI_WRAP; nmfi.lvfi.psz = lpsz; iLen = lstrlen(lpsz); // special case space as the first character if ((iLen == 1) && (*lpsz == ' ')) { if (plv->iFocus != -1) { ListView_OnSetItemState(plv, plv->iFocus, LVIS_SELECTED, LVIS_SELECTED); IncrementSearchString(&plv->is, 0, NULL); } //notify of navigation key usage CCNotifyNavigationKeyUsage(&(plv->ci), UISF_HIDEFOCUS); return; } // Give caller full string in case they want to do something custom if (ListView_IncrementalSearch(plv, iStartFrom, &nmfi, &i)) return; if (iLen > 0 && SameChars(lpsz, lpsz[0])) { // // The user has been typing the same char over and over again. // Switch from incremental search to Windows 3.1 style search. // iStartFrom = plv->iFocus; nmfi.lvfi.psz = lpsz + iLen - 1; } if (i == -1) i = ListView_OnFindItem(plv, iStartFrom, &nmfi.lvfi); if (!ListView_IsValidItemNumber(plv, i)) { i = -1; } TraceMsg(TF_LISTVIEW, "listsearch %d %s %d", (LPTSTR)lpsz, (LPTSTR)lpsz, i); if (i != -1) { ListView_SetFocusSel(plv, i, TRUE, TRUE, FALSE); plv->iMark = i; if (ListView_CancelScrollWait(plv)) ListView_OnEnsureVisible(plv, i, FALSE); } else { // Don't beep on spaces, we use it for selection. IncrementSearchBeep(&plv->is); } //notify of navigation key usage CCNotifyNavigationKeyUsage(&(plv->ci), UISF_HIDEFOCUS); } BOOL SameChars(LPTSTR lpsz, TCHAR c) { while (*lpsz) { if (*lpsz++ != c) return FALSE; } return TRUE; } UINT ListView_OnGetDlgCode(LV* plv, MSG* lpmsg) { return DLGC_WANTARROWS | DLGC_WANTCHARS; } int ListView_ComputeCXItemSize(LV* plv) { int cxItem; cxItem = 16 * plv->cxLabelChar + plv->cxSmIcon; if (cxItem == 0) { cxItem = g_cxBorder; } ASSERT(cxItem != 0); return cxItem; } int ListView_ComputeCYItemSize(LV* plv) { int cyItem; cyItem = max(plv->cyLabelChar, plv->cySmIcon); if (plv->himlState) { cyItem = max(cyItem, plv->cyState); } cyItem += g_cyBorder; ASSERT(cyItem != 0); return cyItem; } void ListView_InvalidateCachedLabelSizes(LV* plv) { int i; ListView_InvalidateTTLastHit(plv, plv->iTTLastHit); // Label wrapping has changed, so we need to invalidate the // size of the items, such that they will be recomputed. // if (!ListView_IsOwnerData(plv)) { for (i = ListView_Count(plv) - 1; i >= 0; i--) { LISTITEM* pitem = ListView_FastGetItemPtr(plv, i); ListView_SetSRecompute(pitem); } } plv->rcView.left = RECOMPUTE; if ((plv->ci.style & LVS_OWNERDRAWFIXED) && ListView_IsReportView(plv)) plv->cyItemSave = ListView_ComputeCYItemSize(plv); else { plv->cyItem = ListView_ComputeCYItemSize(plv); } } void ListView_OnStyleChanging(LV* plv, UINT gwl, LPSTYLESTRUCT pinfo) { if (gwl == GWL_STYLE) { // Don't allow LVS_OWNERDATA to change after creation DWORD stylePreserve = LVS_OWNERDATA; // Don't allow a LVS_EX_REGIONAL listview to change type, since // it must be LVS_ICON // Similarly, HideLabels only works in large icon mode so keep the type. if ((plv->exStyle & LVS_EX_REGIONAL) || ListView_HideLabels(plv)) stylePreserve |= LVS_TYPEMASK; // Preserve the bits that must be preserved pinfo->styleNew ^= (pinfo->styleNew ^ pinfo->styleOld) & stylePreserve; // If we're in group view, then listview must be in autoarrange if (plv->fGroupView) { pinfo->styleNew |= LVS_AUTOARRANGE; } } } WORD MapViewStyle(DWORD style) { if (style == LVS_LIST) return LV_VIEW_LIST; if (style == LVS_SMALLICON) return LV_VIEW_SMALLICON; if (style == LVS_REPORT) return LV_VIEW_DETAILS; return LV_VIEW_ICON; } void ListView_OnStyleChanged(LV* plv, UINT gwl, LPSTYLESTRUCT pinfo) { // Style changed: redraw everything... // // try to do this smartly, avoiding unnecessary redraws if (gwl == GWL_STYLE) { BOOL fRedraw = FALSE, fShouldScroll = FALSE; DWORD changeFlags, styleOld; ListView_DismissEdit(plv, FALSE); changeFlags = plv->ci.style ^ pinfo->styleNew; styleOld = plv->ci.style; // (dli) Setting the small icon width here and only in the case when we go // from large icon view to some other view because of three reasons: // 1. According to chee, we want to set this before we change the style bit in // plv or after we scale. // 2. We don't want to do it after we scale because we want to set the width to // the maximum value so that the items in this listview do not cover each other // 3. we do it from large icon view because large icon view has fixed width for // each item, small icon view width can be scaled. // if ((changeFlags & LVS_TYPEMASK) && (plv->wView == LV_VIEW_ICON)) ListView_ISetColumnWidth(plv, 0, LV_GetNewColWidth(plv, 0, ListView_Count(plv)-1), FALSE); plv->ci.style = pinfo->styleNew; // change our version if (changeFlags & (WS_BORDER | WS_CAPTION | WS_THICKFRAME)) { // the changing of these bits affect the size of the window // but not until after this message is handled // so post ourself a message. PostMessage(plv->ci.hwnd, LVMP_WINDOWPOSCHANGED, 0, 0); } if (changeFlags & LVS_NOCOLUMNHEADER) { if (plv->hwndHdr) { SetWindowBits(plv->hwndHdr, GWL_STYLE, HDS_HIDDEN, (plv->ci.style & LVS_NOCOLUMNHEADER) ? HDS_HIDDEN : 0); fRedraw = TRUE; fShouldScroll = TRUE; } } if (changeFlags & LVS_NOLABELWRAP) { ListView_InvalidateCachedLabelSizes(plv); fShouldScroll = TRUE; fRedraw = TRUE; } if (changeFlags & LVS_TYPEMASK) { WORD wViewOld = plv->wView; plv->wView = MapViewStyle(plv->ci.style & LVS_TYPEMASK); ListView_TypeChange(plv, wViewOld, (BOOL)BOOLIFY(styleOld & LVS_OWNERDRAWFIXED)); fShouldScroll = TRUE; fRedraw = TRUE; } if (changeFlags & LVS_AUTOARRANGE) { if (plv->ci.style & LVS_AUTOARRANGE) { // Turned on. ListView_OnArrange(plv, LVA_DEFAULT); fRedraw = TRUE; } else { // Turned off. Nuke insertmark, because that's not allowed when // auto-arrange is off. LVINSERTMARK lvim = {0}; lvim.cbSize = sizeof(LVINSERTMARK); lvim.iItem = -1; ListView_OnSetInsertMark(plv, &lvim); } } // previously, this was the else to // (changeFlags & LVS_AUTOARRANGE && (plv->ci.style & LVS_AUTOARRANGE)) // I'm not sure that was really the right thing.. if (fShouldScroll) { // Else we would like to make the most important item to still // be visible. So first we will look for a cursorered item // if this fails, we will look for the first selected item, // else we will simply ask for the first item (assuming the // count > 0 // int i; // And make sure the scrollbars are up to date Note this // also updates some variables that some views need ListView_UpdateScrollBars(plv); i = (plv->iFocus >= 0) ? plv->iFocus : ListView_OnGetNextItem(plv, -1, LVNI_SELECTED); if ((i == -1) && (ListView_Count(plv) > 0)) i = 0; if (i != -1) ListView_OnEnsureVisible(plv, i, TRUE); } if (fRedraw) RedrawWindow(plv->ci.hwnd, NULL, NULL, RDW_INVALIDATE | RDW_ERASE); } else if (gwl == GWL_EXSTYLE) { // // If the RTL_MIRROR extended style bit had changed, let's // repaint the control window. // if ((plv->ci.dwExStyle&RTL_MIRRORED_WINDOW) != (pinfo->styleNew&RTL_MIRRORED_WINDOW)) RedrawWindow(plv->ci.hwnd, NULL, NULL, RDW_INVALIDATE | RDW_ERASE); // Save the new ex-style bits plv->ci.dwExStyle = pinfo->styleNew; } // Change of styles also changes tooltip policy, so pop it ListView_PopBubble(plv); } void ListView_TypeChange(LV* plv, WORD wViewOld, BOOL fOwnerDrawFixed) { RECT rc; int i; // // Invalidate all cached string metrics because customdraw clients // may draw differently depending on the type. This happens more // often than you might think, not on purpose, but because apps are // buggy. // if (!ListView_IsOwnerData(plv)) { for (i = 0; i < ListView_Count(plv); i++) { LISTITEM *pitem = ListView_FastGetItemPtr(plv, i); ListView_SetSRecompute(pitem); } } switch (wViewOld) { case LV_VIEW_DETAILS: ShowWindow(plv->hwndHdr, SW_HIDE); if (fOwnerDrawFixed) { // swap cyItem and cyFixed; int temp = plv->cyItem; plv->cyItem = plv->cyItemSave; plv->cyItemSave = temp; } break; default: break; } plv->ptOrigin.x = 0; plv->ptOrigin.y = 0; plv->ptlRptOrigin.x = 0; plv->ptlRptOrigin.y = 0; plv->rcView.left = RECOMPUTE; _ListView_RecomputeEx(plv, NULL, 0, TRUE); // Now handle any special setup needed for the new view switch (plv->wView) { case LV_VIEW_LIST: // We may need to resize the columns ListView_MaybeResizeListColumns(plv, 0, ListView_Count(plv)-1); break; case LV_VIEW_DETAILS: // if it's owner draw fixed, we may have to do funky stuff if (wViewOld != LV_VIEW_DETAILS) { plv->cyItemSave = plv->cyItem; } ListView_RInitialize(plv, FALSE); break; default: break; } ListView_InvalidateTTLastHit(plv, plv->iTTLastHit); GetClientRect(plv->ci.hwnd, &rc); plv->sizeClient.cx = rc.right; plv->sizeClient.cy = rc.bottom; } int ListView_OnHitTest(LV* plv, LV_HITTESTINFO* pinfo) { UINT flags; int x, y; if (!pinfo) return -1; x = pinfo->pt.x; y = pinfo->pt.y; pinfo->iItem = -1; flags = 0; if (x < 0) flags |= LVHT_TOLEFT; else if (x >= plv->sizeClient.cx) flags |= LVHT_TORIGHT; if (y < 0) flags |= LVHT_ABOVE; else if (y >= plv->sizeClient.cy) flags |= LVHT_BELOW; if (flags == 0) { pinfo->iItem = _ListView_ItemHitTest(plv, x, y, &flags, NULL); } pinfo->flags = flags; if (pinfo->iItem >= ListView_Count(plv)) { pinfo->iItem = -1; pinfo->flags = LVHT_NOWHERE; } return pinfo->iItem; } int ScrollAmount(int large, int iSmall, int unit) { return (((large - iSmall) + (unit - 1)) / unit) * unit; } // NOTE: this is duplicated in shell32.dll // // checks to see if we are at the end position of a scroll bar // to avoid scrolling when not needed (avoid flashing) // // in: // code SB_VERT or SB_HORZ // bDown FALSE is up or left // TRUE is down or right BOOL CanScroll(LV* plv, int code, BOOL bDown) { SCROLLINFO si; si.cbSize = sizeof(SCROLLINFO); si.fMask = SIF_RANGE | SIF_PAGE | SIF_POS; if (ListView_GetScrollInfo(plv, code, &si)) { if (bDown) { if (si.nPage) si.nMax -= (si.nPage - 1); return si.nPos < si.nMax; } else { return si.nPos > si.nMin; } } else { return FALSE; } } // detect if we should auto scroll the window // // in: // pt cursor pos in hwnd's client coords // out: // pdx, pdy ammount scrolled in x and y // // REVIEW, this should make sure a certain amount of time has passed // before scrolling. void ScrollDetect(LV* plv, POINT pt, int *pdx, int *pdy) { int dx, dy; *pdx = *pdy = 0; if (!(plv->ci.style & (WS_HSCROLL | WS_VSCROLL))) return; dx = dy = plv->cyIcon / 16; if (ListView_IsReportView(plv)) { if (!plv->fGroupView) // Groupview is always in pixels dy = plv->cyItem; // we scroll in units of items... if (!dx) dx = plv->cxSmIcon; } if (ListView_IsListView(plv)) dx = plv->cxItem; if (!dx) dx = 1; if (!dy) dy = 1; // we need to check if we can scroll before acutally doing it // since the selection rect is adjusted based on how much // we scroll by if (plv->ci.style & WS_VSCROLL) // scroll vertically? { if (pt.y >= plv->sizeClient.cy) { if (CanScroll(plv, SB_VERT, TRUE)) *pdy = ScrollAmount(pt.y, plv->sizeClient.cy, dy); // down } else if (pt.y <= 0) { if (CanScroll(plv, SB_VERT, FALSE)) *pdy = -ScrollAmount(0, pt.y, dy); // up } } if (plv->ci.style & WS_HSCROLL) // horizontally { if (pt.x >= plv->sizeClient.cx) { if (CanScroll(plv, SB_HORZ, TRUE)) *pdx = ScrollAmount(pt.x, plv->sizeClient.cx, dx); // right } else if (pt.x <= 0) { if (CanScroll(plv, SB_HORZ, FALSE)) *pdx = -ScrollAmount(0, pt.x, dx); // left } } // REARCHITECT: this will potentially scroll outside the bounds of the // listview. we should bound the scroll amount in CanScroll() // or ScrollAmount(). if (*pdx || *pdy) { ListView_ValidateScrollParams(plv, pdx, pdy); } } #define swap(pi1, pi2) {int i = *(pi1) ; *(pi1) = *(pi2) ; *(pi2) = i ;} void OrderRect(RECT *prc) { if (prc->left > prc->right) swap(&prc->left, &prc->right); if (prc->bottom < prc->top) swap(&prc->bottom, &prc->top); } // in: // x, y starting point in client coords #define SCROLL_FREQ (GetDoubleClickTime()/2) // 1/5 of a second between scrolls BOOL ShouldScroll(LV* plv, LPPOINT ppt, LPRECT lprc) { ASSERT(ppt); if (plv->ci.style & WS_VSCROLL) { if (ppt->y >= lprc->bottom) { if (CanScroll(plv, SB_VERT, TRUE)) return TRUE; } else if (ppt->y <= lprc->top) { if (CanScroll(plv, SB_VERT, FALSE)) return TRUE; } } if (plv->ci.style & WS_HSCROLL) { if (ppt->x >= lprc->right) { if (CanScroll(plv, SB_HORZ, TRUE)) return TRUE; } else if (ppt->x <= lprc->left) { if (CanScroll(plv, SB_HORZ, FALSE)) return TRUE; } } return FALSE; } BOOL DrawFocusRectClip(HDC hdc, CONST RECT * prc, CONST RECT * prcClip) { RECT rc; IntersectRect(&rc, prc, prcClip); return DrawFocusRect(hdc, &rc); } // Listview is "Alpha Capable" if: // Colors >= 16bpp (Needed for alpha) // The Listview is double buffered (Needed for flicker) // The use has "Show window contents while dragging" (Needed to turn off on slow machines) // NOTE: g_fDragFullWindows is turned off in comctl32 when running a remote session BOOL ListView_IsAlphaMarqueeCapable(LV* plv) { BOOL fAlphaCapable = FALSE; if (ListView_IsDoubleBuffer(plv)) { if (AreAllMonitorsAtLeast(16)) { fAlphaCapable = plv->fListviewAlphaSelect; } } return fAlphaCapable; } void ListView_DragSelect(LV *plv, int x, int y) { RECT rc, rcWindow, rcOld, rcUnion, rcTemp2, rcClip; POINT pt; MSG32 msg32; HDC hdc; HWND hwnd = plv->ci.hwnd; int i, iEnd, dx, dy; BOOL bInOld, bInNew = FALSE, bLocked = FALSE; DWORD dwTime, dwNewTime; HRGN hrgnUpdate = NULL, hrgnLV = NULL; BOOL fAlphaMarquee = ListView_IsAlphaMarqueeCapable(plv); rc.left = rc.right = x; rc.top = rc.bottom = y; rcOld = rc; UpdateWindow(plv->ci.hwnd); if (plv->exStyle & LVS_EX_REGIONAL) { if ((hrgnUpdate = CreateRectRgn(0,0,0,0)) && (hrgnLV = CreateRectRgn(0,0,0,0)) && (LockWindowUpdate(GetParent(hwnd)))) { hdc = GetDCEx(hwnd, NULL, DCX_PARENTCLIP | DCX_LOCKWINDOWUPDATE); bLocked = TRUE; } else { goto BailOut; } } else { hdc = GetDC(hwnd); } SetCapture(hwnd); if (fAlphaMarquee) { plv->flags |= LVF_MARQUEE; plv->rcMarquee = rc; InvalidateRect(plv->ci.hwnd, &plv->rcMarquee, TRUE); } else { DrawFocusRect(hdc, &rc); } GetClientRect(hwnd, &rcClip); GetWindowRect(hwnd, &rcWindow); dwTime = GetTickCount(); for (;;) { // WM_CANCELMODE messages will unset the capture, in that // case I want to exit this loop if (GetCapture() != hwnd) { break; } if (!PeekMessage32(&msg32, NULL, 0, 0, PM_REMOVE, TRUE)) { // if the cursor is outside of the window rect // we need to generate messages to make autoscrolling // keep going if (!PtInRect(&rcWindow, msg32.pt) && ShouldScroll(plv, &msg32.pt, &rcWindow)) { SetCursorPos(msg32.pt.x, msg32.pt.y); } else { WaitMessage(); } continue; } // See if the application wants to process the message... if (CallMsgFilter32(&msg32, MSGF_COMMCTRL_DRAGSELECT, TRUE) != 0) continue; switch (msg32.message) { case WM_LBUTTONUP: case WM_RBUTTONUP: case WM_LBUTTONDOWN: case WM_MBUTTONDOWN: case WM_MBUTTONUP: case WM_RBUTTONDOWN: CCReleaseCapture(&plv->ci); goto EndOfLoop; case WM_TIMER: if (msg32.wParam != IDT_MARQUEE) goto DoDefault; // else fall through case WM_MOUSEMOVE: { int dMax = -1; pt = msg32.pt; ScreenToClient(hwnd, &pt); dwNewTime = GetTickCount(); // if (1 || (dwNewTime - dwTime) > SCROLL_FREQ) // { dwTime = dwNewTime; // reset scroll timer ScrollDetect(plv, pt, &dx, &dy); // } // else // { // dx = dy = 0; // } //SetTimer(plv->ci.hwnd, IDT_MARQUEE, SCROLL_FREQ, NULL); y -= dy; // scroll up/down x -= dx; // scroll left/right rc.left = x; rc.top = y; rc.right = pt.x; rc.bottom = pt.y; // clip drag rect to the window // if (rc.right > rcClip.right) rc.right = rcClip.right; if (rc.right < rcClip.left) rc.right = rcClip.left; if (rc.bottom > rcClip.bottom) rc.bottom = rcClip.bottom; if (rc.bottom < rcClip.top) rc.bottom = rcClip.top; OrderRect(&rc); if (EqualRect(&rc, &rcOld)) break; // move the old rect if (!fAlphaMarquee) { DrawFocusRect(hdc, &rcOld); // erase old } if (dx || dy) ListView_OnScroll(plv, dx, dy); OffsetRect(&rcOld, -dx, -dy); // // For Report and List view, we can speed things up by // only searching through those items that are visible. We // use the hittest to calculate the first item to paint. // REARCHITECT:: We are using state specific info here... // UnionRect(&rcUnion, &rc, &rcOld); if (ListView_IsReportView(plv) && !plv->fGroupView) { i = (int)((plv->ptlRptOrigin.y + rcUnion.top - plv->yTop) / plv->cyItem); iEnd = (int)((plv->ptlRptOrigin.y + rcUnion.bottom - plv->yTop) / plv->cyItem) + 1; } else if (ListView_IsListView(plv)) { i = ((plv->xOrigin + rcUnion.left)/ plv->cxItem) * plv->cItemCol + rcUnion.top / plv->cyItem; iEnd = ((plv->xOrigin + rcUnion.right)/ plv->cxItem) * plv->cItemCol + rcUnion.bottom / plv->cyItem + 1; } else { if (ListView_IsOwnerData(plv)) { ListView_CalcMinMaxIndex(plv, &rcUnion, &i, &iEnd); } else { i = 0; iEnd = ListView_Count(plv); } } // make sure our endpoint is in range. if (iEnd > ListView_Count(plv)) iEnd = ListView_Count(plv); if (i < 0) i = 0; if (ListView_IsOwnerData(plv) && (i < iEnd)) { ListView_NotifyCacheHint(plv, i, iEnd-1); } if (bInNew && !(msg32.wParam & (MK_CONTROL | MK_SHIFT))) { plv->iMark = -1; } for (; i < iEnd; i++) { RECT dummy; ListView_GetRects(plv, i, QUERY_DEFAULT, NULL, NULL, NULL, &rcTemp2); // don't do this infaltion if we're in report&full row mode // in that case, just touching is good enough if (!(ListView_IsReportView(plv) && ListView_FullRowSelect(plv))) { int cxInflate = (rcTemp2.right - rcTemp2.left) / 4; if (ListView_IsListView(plv)) { cxInflate = min(cxInflate, plv->cxSmIcon); } InflateRect(&rcTemp2, -cxInflate, -(rcTemp2.bottom - rcTemp2.top) / 4); } bInOld = (IntersectRect(&dummy, &rcOld, &rcTemp2) != 0); bInNew = (IntersectRect(&dummy, &rc, &rcTemp2) != 0); if (msg32.wParam & MK_CONTROL) { if (bInOld != bInNew) { ListView_ToggleSelection(plv, i); } } else { // was there a change? if (bInOld != bInNew) { ListView_OnSetItemState(plv, i, bInOld ? 0 : LVIS_SELECTED, LVIS_SELECTED); } // if no alternate keys are down.. set the mark to // the item furthest from the cursor if (bInNew && !(msg32.wParam & (MK_CONTROL | MK_SHIFT))) { int dItem; dItem = (rcTemp2.left - pt.x) * (rcTemp2.left - pt.x) + (rcTemp2.top - pt.y) * (rcTemp2.top - pt.y); // if it's further away, set this as the mark //DebugMsg(TF_LISTVIEW, "dItem = %d, dMax = %d", dItem, dMax); if (dItem > dMax) { //DebugMsg(TF_LISTVIEW, "taking dItem .. iMark = %d", i); dMax = dItem; plv->iMark = i; } } } } if (fAlphaMarquee) { RECT rcInvalid; UnionRect(&rcInvalid, &rcOld, &rc); InflateRect(&rcInvalid, 1, 1); plv->flags |= LVF_MARQUEE; plv->rcMarquee = rc; InvalidateRect(plv->ci.hwnd, &rcInvalid, TRUE); } //DebugMsg(TF_LISTVIEW, "Final iMark = %d", plv->iMark); if (bLocked) { if (GetUpdateRgn(plv->ci.hwnd, hrgnUpdate, FALSE) > NULLREGION) { ValidateRect(plv->ci.hwnd, NULL); GetWindowRgn(plv->ci.hwnd, hrgnLV); CombineRgn(hrgnUpdate, hrgnUpdate, hrgnLV, RGN_AND); SelectClipRgn(hdc, hrgnUpdate); SendMessage(plv->ci.hwnd, WM_PRINTCLIENT, (WPARAM)hdc, 0); SelectClipRgn(hdc, NULL); } } else { UpdateWindow(plv->ci.hwnd); // make selection draw } if (!fAlphaMarquee) { DrawFocusRect(hdc, &rc); } rcOld = rc; break; } case WM_KEYDOWN: switch (msg32.wParam) { case VK_ESCAPE: ListView_DeselectAll(plv, -1); goto EndOfLoop; } case WM_CHAR: case WM_KEYUP: // don't process thay keyboard stuff during marquee break; default: // don't process mouse wheel stuff if (msg32.message == g_msgMSWheel) break; DoDefault: TranslateMessage32(&msg32, TRUE); DispatchMessage32(&msg32, TRUE); } } EndOfLoop: plv->flags &= ~LVF_MARQUEE; if (fAlphaMarquee) { InvalidateRect(plv->ci.hwnd, &rcOld, TRUE); } else { DrawFocusRect(hdc, &rcOld); // erase old } ReleaseDC(hwnd, hdc); BailOut: if (hrgnUpdate) DeleteObject(hrgnUpdate); if (hrgnLV) DeleteObject(hrgnLV); if (bLocked) LockWindowUpdate(NULL); } #define SHIFT_DOWN(keyFlags) (keyFlags & MK_SHIFT) #define CONTROL_DOWN(keyFlags) (keyFlags & MK_CONTROL) #define RIGHTBUTTON(keyFlags) (keyFlags & MK_RBUTTON) void ListView_ButtonSelect(LV* plv, int iItem, UINT keyFlags, BOOL bSelected) { if (SHIFT_DOWN(keyFlags)) { ListView_SelectRangeTo(plv, iItem, !CONTROL_DOWN(keyFlags)); ListView_SetFocusSel(plv, iItem, TRUE, FALSE, FALSE); } else if (!CONTROL_DOWN(keyFlags)) { ListView_SetFocusSel(plv, iItem, TRUE, !bSelected, FALSE); if (!RIGHTBUTTON(keyFlags) && bSelected && ListView_IsSimpleSelect(plv)) { ListView_HandleStateIconClick(plv, iItem); } } } void ListView_HandleStateIconClick(LV* plv, int iItem) { int iState = ListView_OnGetItemState(plv, iItem, LVIS_STATEIMAGEMASK); iState = STATEIMAGEMASKTOINDEX(iState) -1; iState++; iState %= ImageList_GetImageCount(plv->himlState); iState++; ListView_OnSetItemState(plv, iItem, INDEXTOSTATEIMAGEMASK(iState), LVIS_STATEIMAGEMASK); } BOOL ListView_RBeginMarquee(LV* plv, int x, int y, LPLVHITTESTINFO plvhti) { if (ListView_FullRowSelect(plv) && ListView_IsReportView(plv) && !(plv->ci.style & LVS_SINGLESEL) && !ListView_OwnerDraw(plv) && plvhti->iSubItem == 0) { // can only begin marquee in column 0. if (plvhti->flags == LVHT_ONITEM) { return TRUE; } } return FALSE; } void ListView_HandleMouse(LV* plv, BOOL fDoubleClick, int x, int y, UINT keyFlags, BOOL bMouseWheel) { LV_HITTESTINFO ht; NMITEMACTIVATE nm; int iItem, click, drag; BOOL bSelected, fHadFocus, fNotifyReturn = FALSE; BOOL fActive; HWND hwnd = plv->ci.hwnd; if (plv->fButtonDown) return; plv->fButtonDown = TRUE; if (plv->exStyle & LVS_EX_ONECLICKACTIVATE && plv->fOneClickHappened && plv->fOneClickOK) { KillTimer(plv->ci.hwnd, IDT_ONECLICKHAPPENED); plv->fOneClickHappened = FALSE; CCSendNotify(&plv->ci, LVN_ITEMACTIVATE, &(plv->nmOneClickHappened.hdr)); if (!IsWindow(hwnd)) return; } fHadFocus = (GetFocus() == plv->ci.hwnd); click = RIGHTBUTTON(keyFlags) ? NM_RCLICK : NM_CLICK; drag = RIGHTBUTTON(keyFlags) ? LVN_BEGINRDRAG : LVN_BEGINDRAG; fActive = ChildOfActiveWindow(plv->ci.hwnd) || fShouldFirstClickActivate() || ChildOfDesktop(plv->ci.hwnd); TraceMsg(TF_LISTVIEW, "ListView_OnButtonDown %d", fDoubleClick); SetCapture(plv->ci.hwnd); plv->ptCapture.x = x; plv->ptCapture.y = y; if (!ListView_DismissEdit(plv, FALSE) && GetCapture() != plv->ci.hwnd) goto EndButtonDown; CCReleaseCapture(&plv->ci); // REVIEW: right button implies no shift or control stuff // Single selection style also implies no modifiers //if (RIGHTBUTTON(keyFlags) || (plv->ci.style & LVS_SINGLESEL)) if ((plv->ci.style & LVS_SINGLESEL)) keyFlags &= ~(MK_SHIFT | MK_CONTROL); ht.pt.x = x; ht.pt.y = y; iItem = ListView_OnSubItemHitTest(plv, &ht); if (ht.iSubItem != 0) { // if we're not in full row select, // hitting on a subitem is like hitting on nowhere // also, in win95, ownerdraw fixed effectively had full row select if (!ListView_FullRowSelect(plv) && !(plv->ci.style & LVS_OWNERDRAWFIXED)) { iItem = -1; ht.flags = LVHT_NOWHERE; } } nm.iItem = iItem; nm.iSubItem = ht.iSubItem; nm.uChanged = 0; nm.ptAction.x = x; nm.ptAction.y = y; nm.uKeyFlags = GetLVKeyFlags(); // FProt Profesional assumed that if the notification structure pointer + 14h bytes // had a value 2 that it was a displayinfo structure and they then used offset +2c as lparam... nm.uNewState = 0; plv->iNoHover = iItem; bSelected = (iItem >= 0) && ListView_OnGetItemState(plv, iItem, LVIS_SELECTED); if (fDoubleClick) { // Cancel any name editing that might happen. ListView_CancelPendingEdit(plv); KillTimer(plv->ci.hwnd, IDT_SCROLLWAIT); if (ht.flags & LVHT_NOWHERE) { // this would have been done in the first click in win95 except // now we blow off the first click on focus change if (!SHIFT_DOWN(keyFlags) && !CONTROL_DOWN(keyFlags)) ListView_DeselectAll(plv, -1); } click = RIGHTBUTTON(keyFlags) ? NM_RDBLCLK : NM_DBLCLK ; if (CCSendNotify(&plv->ci, click, &nm.hdr)) goto EndButtonDown; /// some (comdlg32 for example) destroy on double click // we need to bail if that happens because plv is no longer valid if (!IsWindow(hwnd)) return; if (click == NM_DBLCLK) { // these shift control flags are to mirror when we don't send out the activate on the single click, // but are in the oneclick activate mode (see below) if (ht.flags & (LVHT_ONITEMLABEL | LVHT_ONITEMICON)) { // possible scenarios below: // 1) we're using classic windows style so double click => launch // 2) we're using single click activate // a) shift is down and item is selected => launch // this implies that the first click selected it // b) control is down => launch // the first click toggled the selection so if the item was // the only item selected and we double clicked on it // the first click deselects it and no item is selected // so nothing will be launched - this is win95 behavior if (!(plv->exStyle & LVS_EX_ONECLICKACTIVATE && plv->fOneClickOK) || (plv->exStyle & LVS_EX_ONECLICKACTIVATE && plv->fOneClickOK && (SHIFT_DOWN(keyFlags) || CONTROL_DOWN(keyFlags)))) { CCSendNotify(&plv->ci, LVN_ITEMACTIVATE, &nm.hdr); } } // Double-click on checkbox state icon cycles it just like single click else if ((ht.flags & LVHT_ONITEMSTATEICON) && ListView_CheckBoxes(plv)) { ListView_HandleStateIconClick(plv, iItem); } } if (!IsWindow(hwnd)) return; goto EndButtonDown; } if (ht.flags & (LVHT_ONITEMLABEL | LVHT_ONITEMICON)) { // if it wasn't selected, we're about to select it... play // a little ditty for us... CCPlaySound(c_szSelect); if (!RIGHTBUTTON(keyFlags) || (!CONTROL_DOWN(keyFlags) && !SHIFT_DOWN(keyFlags))) ListView_ButtonSelect(plv, iItem, keyFlags, bSelected); // handle full row select // If single-select listview, disable marquee selection. // // Careful - CheckForDragBegin yields and the app may have // destroyed the item we were thinking about dragging! // if (!bMouseWheel && CheckForDragBegin(plv->ci.hwnd, x, y)) { // should we do a marquee? if (ListView_RBeginMarquee(plv, x, y, &ht) && !CCSendNotify(&plv->ci, LVN_MARQUEEBEGIN, &nm.hdr)) { ListView_DragSelect(plv, x, y); fNotifyReturn = !CCSendNotify(&plv->ci, click, &nm.hdr); } else { // Before we start dragging, make it sure that it is // selected and has the focus. ListView_SetFocusSel(plv, iItem, TRUE, FALSE, FALSE); if (!SHIFT_DOWN(keyFlags)) plv->iMark = iItem; // Then, we need to update the window before start dragging // to show the selection chagne. UpdateWindow(plv->ci.hwnd); // Remember which item we're dragging, as it affects ListView_OnInsertMarkHitTest plv->iDrag = iItem; CCSendNotify(&plv->ci, drag, &nm.hdr); plv->iDrag = -1; goto EndButtonDown; } } // CheckForDragBegin yields, so revalidate before continuing else if (IsWindow(hwnd)) { // button came up and we are not dragging if (!RIGHTBUTTON(keyFlags)) { if (CONTROL_DOWN(keyFlags)) { // do this on the button up so that ctrl-dragging a range // won't toggle the select. if (SHIFT_DOWN(keyFlags)) ListView_SetFocusSel(plv, iItem, FALSE, FALSE, FALSE); else { ListView_SetFocusSel(plv, iItem, TRUE, FALSE, TRUE); } } } if (!SHIFT_DOWN(keyFlags)) plv->iMark = iItem; if (!ListView_SetFocus(plv->ci.hwnd)) // activate this window return; // now do the deselect stuff if (!SHIFT_DOWN(keyFlags) && !CONTROL_DOWN(keyFlags) && !RIGHTBUTTON(keyFlags)) { ListView_DeselectAll(plv, iItem); if ((ht.flags & LVHT_ONITEMLABEL) && bSelected && !(plv->exStyle & (LVS_EX_ONECLICKACTIVATE|LVS_EX_TWOCLICKACTIVATE))) { // doing this check for ownerdrawfixed is for compatability. // we don't want to go into edit mode if the user just happened to click // to this window when a different one had focus, // but ms hammer relied upon the notification being sent (and we // don't go into edit mode anyways for ownerdraw) if (fHadFocus || (plv->ci.style & LVS_OWNERDRAWFIXED)) { // Click on item label. It was selected and // no modifier keys were pressed and no drag operation // So setup for name edit mode. Still need to wait // to make sure user is not doing double click. // ListView_SetupPendingNameEdit(plv); } } } fNotifyReturn = !CCSendNotify(&plv->ci, click, &nm.hdr); if (!IsWindow(hwnd)) return; if (plv->exStyle & (LVS_EX_ONECLICKACTIVATE|LVS_EX_TWOCLICKACTIVATE)) { if (!RIGHTBUTTON(keyFlags)) { // We don't ItemActivate within one double-click time of creating // this listview. This is a common occurence for people used to // double-clicking. The first click pops up a new window which // receives the second click and ItemActivates the item... // if ((plv->exStyle & LVS_EX_ONECLICKACTIVATE && plv->fOneClickOK) || bSelected) { if (fActive) { // condition: if we're in a single click activate mode // don't launch if control or shift keys are pressed BOOL bCond = plv->exStyle & LVS_EX_ONECLICKACTIVATE && !CONTROL_DOWN(keyFlags) && !SHIFT_DOWN(keyFlags); if ((bSelected && plv->exStyle & LVS_EX_TWOCLICKACTIVATE) || (bCond && !g_bUseDblClickTimer)) { CCSendNotify(&plv->ci, LVN_ITEMACTIVATE, &nm.hdr); if (!IsWindow(hwnd)) return; } else if (bCond && g_bUseDblClickTimer) { plv->fOneClickHappened = TRUE; plv->nmOneClickHappened = nm; SetTimer(plv->ci.hwnd, IDT_ONECLICKHAPPENED, GetDoubleClickTime(), NULL); } } } } } } else { // IsWindow() failed. Bail. return; } } else if (ht.flags & LVHT_ONITEMSTATEICON) { // Should activate window and send notificiation to parent... if (!ListView_SetFocus(plv->ci.hwnd)) // activate this window return; fNotifyReturn = !CCSendNotify(&plv->ci, click, &nm.hdr); if (fNotifyReturn && ListView_CheckBoxes(plv)) { ListView_HandleStateIconClick(plv, iItem); } } else if (ht.flags & LVHT_NOWHERE) { if (!ListView_SetFocus(plv->ci.hwnd)) // activate this window return; // If single-select listview, disable marquee selection. if (!(plv->ci.style & LVS_SINGLESEL) && CheckForDragBegin(plv->ci.hwnd, x, y) && !CCSendNotify(&plv->ci, LVN_MARQUEEBEGIN, &nm.hdr)) { if (!SHIFT_DOWN(keyFlags) && !CONTROL_DOWN(keyFlags)) ListView_DeselectAll(plv, -1); ListView_DragSelect(plv, x, y); fNotifyReturn = !CCSendNotify(&plv->ci, click, &nm.hdr); } else if (IsWindow(hwnd)) { // if we didn't have focus and aren't showing selection always, // make the first click just set focus BOOL fDoFirstClickSelection = (fHadFocus || plv->ci.style & LVS_SHOWSELALWAYS || CONTROL_DOWN(keyFlags) || SHIFT_DOWN(keyFlags) || RIGHTBUTTON(keyFlags)); if (fDoFirstClickSelection && fActive) { if (!SHIFT_DOWN(keyFlags) && !CONTROL_DOWN(keyFlags)) ListView_DeselectAll(plv, -1); fNotifyReturn = !CCSendNotify(&plv->ci, click, &nm.hdr); } } else { // IsWindow() failed. Bail. return; } } // re-check the key state so we don't get confused by multiple clicks // this needs to check the GetKeyState stuff only when we've gone into // a modal loop waiting for the rbutton up. if (fNotifyReturn && (click == NM_RCLICK)) // && (GetKeyState(VK_RBUTTON)>=0)) { POINT pt = { x, y }; ClientToScreen(plv->ci.hwnd, &pt); FORWARD_WM_CONTEXTMENU(plv->ci.hwnd, plv->ci.hwnd, pt.x, pt.y, SendMessage); } EndButtonDown: if (IsWindow(hwnd)) plv->fButtonDown = FALSE; } void ListView_OnButtonDown(LV* plv, BOOL fDoubleClick, int x, int y, UINT keyFlags) { ListView_HandleMouse(plv, fDoubleClick, x, y, keyFlags, FALSE); } BOOL ListView_CancelPendingTimer(LV* plv, UINT fFlags, int idTimer) { if (plv->flags & fFlags) { KillTimer(plv->ci.hwnd, idTimer); plv->flags &= ~fFlags; return TRUE; } return FALSE; } // // ListView_OnTimer: // process the WM_TIMER message. If the timer id is thta // of the name editing, we should then start the name editing mode. // void ListView_OnTimer(LV* plv, UINT id) { KillTimer(plv->ci.hwnd, id); if (id == IDT_NAMEEDIT) { // Kill the timer as we wont need any more messages from it. if (ListView_CancelPendingEdit(plv)) { // And start name editing mode. if (!ListView_OnEditLabel(plv, plv->iFocus, NULL)) { ListView_DismissEdit(plv, FALSE); ListView_SetFocusSel(plv, plv->iFocus, TRUE, TRUE, FALSE); } } } else if (id == IDT_SCROLLWAIT) { if (ListView_CancelScrollWait(plv)) { ListView_OnEnsureVisible(plv, plv->iFocus, TRUE); } } else if (id == IDT_ONECLICKOK) { plv->fOneClickOK = TRUE; } else if (id == IDT_ONECLICKHAPPENED) { //if (!g_bUseDblClickTimer) //{ //// EnableWindow(plv->ci.hwnd, TRUE); // SetWindowBits(plv->ci.hwnd, GWL_STYLE, WS_DISABLED, 0); // plv->fOneClickHappened = FALSE; //} // check the bit just in case they double-clicked //else if (plv->fOneClickHappened) { plv->fOneClickHappened = FALSE; CCSendNotify(&plv->ci, LVN_ITEMACTIVATE, &(plv->nmOneClickHappened.hdr)); } } else if (id == IDT_TRACKINGTIP) { // Display keyboard nav tracking tooltip popups if (ListView_IsKbdTipTracking(plv)) // Item requires tracking popup { // Ensure index is still valid if (ListView_IsValidItemNumber(plv, plv->iTracking)) { TOOLINFO ti = {0}; ti.cbSize = sizeof(TOOLINFO); ti.hwnd = plv->ci.hwnd; // Cancel previous SendMessage(plv->hwndToolTips, TTM_TRACKACTIVATE, FALSE, (LPARAM)&ti); // Switch ListView's tooltip window to "tracking" (manual) mode SendMessage(plv->hwndToolTips, TTM_GETTOOLINFO, 0, (LPARAM)&ti); ti.uFlags |= TTF_TRACK; SendMessage(plv->hwndToolTips, TTM_SETTOOLINFO, 0, (LPARAM)&ti); // Activate and establish size SendMessage(plv->hwndToolTips, TTM_TRACKACTIVATE, TRUE, (LPARAM)&ti); } else { // Index was invalid (ListView set of items changed), tip track cancel, no popup plv->iTracking = LVKTT_NOTRACK; } } } } // // ListView_SetupPendingNameEdit: // Sets up a timer to begin name editing at a delayed time. This // will allow the user to double click on the already selected item // without going into name editing mode, which is especially important // in those views that only show a small icon. // void ListView_SetupPendingNameEdit(LV* plv) { SetTimer(plv->ci.hwnd, IDT_NAMEEDIT, GetDoubleClickTime(), NULL); plv->flags |= LVF_NMEDITPEND; } void ListView_OnHVScroll(LV* plv, UINT code, int pos, int sb) { int iScrollCount = 0; SCROLLINFO si; si.cbSize = sizeof(SCROLLINFO); si.fMask = SIF_TRACKPOS; // if we're in 32bits, don't trust the pos since it's only 16bit's worth if (ListView_GetScrollInfo(plv, sb, &si)) pos = (int)si.nTrackPos; ListView_DismissEdit(plv, FALSE); _ListView_OnScroll(plv, code, pos, sb); switch (code) { case SB_PAGELEFT: case SB_PAGERIGHT: if (plv->iScrollCount < SMOOTHSCROLLLIMIT) plv->iScrollCount += 3; break; case SB_LINELEFT: case SB_LINERIGHT: if (plv->iScrollCount < SMOOTHSCROLLLIMIT) plv->iScrollCount++; break; case SB_ENDSCROLL: plv->iScrollCount = 0; break; } } void ListView_OnVScroll(LV* plv, HWND hwndCtl, UINT code, int pos) { ListView_OnHVScroll(plv, code, pos, SB_VERT); } void ListView_OnHScroll(LV* plv, HWND hwndCtl, UINT code, int pos) { ListView_OnHVScroll(plv, code, pos, SB_HORZ); } int ListView_ValidateOneScrollParam(LV* plv, int iDirection, int dx) { SCROLLINFO si; si.cbSize = sizeof(SCROLLINFO); si.fMask = SIF_PAGE | SIF_RANGE | SIF_POS; if (!ListView_GetScrollInfo(plv, iDirection, &si)) return 0; if (si.nPage) si.nMax -= (si.nPage - 1); si.nPos += dx; if (si.nPos < si.nMin) { dx += (int)(si.nMin - si.nPos); } else if (si.nPos > si.nMax) { dx -= (int)(si.nPos - si.nMax); } return dx; } BOOL ListView_ValidateScrollParams(LV* plv, int * pdx, int *pdy) { int dx = *pdx; int dy = *pdy; if (plv->ci.style & LVS_NOSCROLL) return FALSE; if (ListView_IsListView(plv)) { ListView_MaybeResizeListColumns(plv, 0, ListView_Count(plv)-1); #ifdef COLUMN_VIEW if (dx < 0) dx = (dx - (plv->cxItem - 1)) / plv->cxItem; else dx = (dx + (plv->cxItem - 1)) / plv->cxItem; if (dy) return FALSE; #else if (dy < 0) dy = (dy - (plv->cyItem - 1)) / plv->cyItem; else dy = (dy + (plv->cyItem - 1)) / plv->cyItem; if (dx) return FALSE; #endif } else if (ListView_IsReportView(plv)) { // // Note: This function expects that dy is in number of lines // and we are working with pixels so do a conversion use some // rounding up and down to make it right if (dy > 0) dy = (dy + plv->cyItem/2) / plv->cyItem; else dy = (dy - plv->cyItem/2) / plv->cyItem; } if (dy) { dy = ListView_ValidateOneScrollParam(plv, SB_VERT, dy); if (ListView_IsReportView(plv) #ifndef COLUMN_VIEW || ListView_IsListView(plv) #endif ) { // convert back to pixels dy *= plv->cyItem; } *pdy = dy; } if (dx) { dx = ListView_ValidateOneScrollParam(plv, SB_HORZ, dx); #ifdef COLUMN_VIEW if (ListView_IsListView(plv)) { dx *= plv->cxItem; } #endif *pdx = dx; } return TRUE; } BOOL ListView_SendScrollNotify(LV* plv, BOOL fBegin, int dx, int dy) { NMLVSCROLL nm; nm.dx = dx; nm.dy = dy; return !CCSendNotify(&plv->ci, fBegin ? LVN_BEGINSCROLL : LVN_ENDSCROLL, &nm.hdr); } BOOL ListView_OnScrollSelectSmooth(LV* plv, int dx, int dy, UINT uSmooth) { if (plv->ci.style & LVS_NOSCROLL) return FALSE; #ifdef DEBUG // If we try and scroll an illegal amount then ListView_IScroll2_SmoothScroll // will offset ptOrigin incorrectly (it doesn't check min/max range) which then // mucks up hit testing and insert marks if (ListView_IsIScrollView(plv)) { int dxTmp = dx; int dyTmp = dy; ASSERT(ListView_ValidateScrollParams(plv, &dxTmp, &dyTmp) && dxTmp == dx && dyTmp == dy); } #endif if (ListView_IsListView(plv)) { // Scale pixel count to column count // #ifdef COLUMN_VIEW if (dx < 0) dx -= plv->cxItem - 1; else dx += plv->cxItem - 1; dx = dx / plv->cxItem; if (dy) return FALSE; #else if (dy < 0) dy -= plv->cyItem - 1; else dy += plv->cyItem - 1; dy = dy / plv->cyItem; if (dx) return FALSE; #endif } else if (ListView_IsReportView(plv) && !plv->fGroupView) { // // Note: This function expects that dy is in number of lines // and we are working with pixels so do a conversion use some // rounding up and down to make it right if (dy > 0) dy = (dy + plv->cyItem/2) / plv->cyItem; else dy = (dy - plv->cyItem/2) / plv->cyItem; } ListView_SendScrollNotify(plv, TRUE, dx, dy); _ListView_Scroll2(plv, dx, dy, uSmooth); ListView_SendScrollNotify(plv, FALSE, dx, dy); ListView_UpdateScrollBars(plv); return TRUE; } BOOL ListView_OnScroll(LV* plv, int dx, int dy) { return ListView_OnScrollSelectSmooth(plv, dx, dy, 0); } #ifdef DEBUG BOOL ListView_ValidatercView(LV* plv, RECT* prcView, BOOL fRecalcDone) { BOOL fRet = prcView->left != RECOMPUTE ? TRUE : !fRecalcDone; // hitting this assert is only valuable if there's a manual repro, which never happens in stress #ifdef FULL_DEBUG if (!ListView_IsOwnerData(plv) && ListView_IsIScrollView(plv) && !(plv->fGroupView && plv->hdpaGroups) && ListView_RedrawEnabled(plv)) { RECT rcViewTmp; fRet = ListView_ICalcViewRect(plv, TRUE, &rcViewTmp); if (fRet) { ASSERT(prcView->left != RECOMPUTE); fRet = IsEqualRect(rcViewTmp, *prcView); } else { fRet = !fRecalcDone; } } #endif return fRet; } BOOL ListView_ValidateScrollPositions(LV* plv, RECT* prcClient) { BOOL fRet = TRUE; // hitting this assert is only valuable if there's a manual repro, which never happens in stress #ifdef FULL_DEBUG // if we're in ListView_FixIScrollPositions, then it will fix up the scroll positions when we unwind if (ListView_IsIScrollView(plv) && (!plv->fInFixIScrollPositions) && ListView_RedrawEnabled(plv)) { if (!(plv->ci.style & LVS_NOSCROLL)) { // if we don't have a client rect there's no way to validate anything, assume everything will be recomputed later RECT rcClient; if (!prcClient) { if (plv->rcView.left != RECOMPUTE) { ListView_GetStyleAndClientRectGivenViewRect(plv, &plv->rcView, &rcClient); prcClient = &rcClient; } } if (prcClient) { if (fRet) { if (RECTWIDTH(*prcClient) < RECTWIDTH(plv->rcView)) { fRet = (plv->rcView.left <= plv->ptOrigin.x) && (plv->ptOrigin.x+RECTWIDTH(*prcClient) <= plv->rcView.right); } else { fRet = (plv->ptOrigin.x <= plv->rcView.left) && (plv->rcView.right <= plv->ptOrigin.x+RECTWIDTH(*prcClient)); } } if (fRet) { if (RECTHEIGHT(*prcClient) < RECTHEIGHT(plv->rcView)) { fRet = (plv->rcView.top <= plv->ptOrigin.y) && (plv->ptOrigin.y+RECTHEIGHT(*prcClient) <= plv->rcView.bottom); } else { fRet = (plv->ptOrigin.y <= plv->rcView.top) && (plv->rcView.bottom <= plv->ptOrigin.y+RECTHEIGHT(*prcClient)); } } } } else { fRet = (plv->ptOrigin.x == 0) && (plv->ptOrigin.y == 0); } } #endif return fRet; } #endif BOOL ListView_OnEnsureVisible(LV* plv, int i, BOOL fPartialOK) { RECT rcBounds; RECT rc; RECT rcClient; int dx, dy; if (!ListView_IsValidItemNumber(plv, i) || plv->ci.style & LVS_NOSCROLL) return FALSE; // we need to do this again inside because some callers don't do it. // other callers that do this need to do it outside so that // they can know not to call us if there's not wait pending ListView_CancelScrollWait(plv); if (ListView_IsReportView(plv)) return ListView_ROnEnsureVisible(plv, i, fPartialOK); ListView_GetRects(plv, i, QUERY_DEFAULT, &rc, NULL, &rcBounds, NULL); if (plv->fGroupView) { LISTITEM* pitem = ListView_GetItemPtr(plv, i); if (pitem) { LISTGROUP* pgrp = ListView_FindFirstVisibleGroup(plv); if (pitem->pGroup == pgrp && pgrp) { rcBounds.top -= LISTGROUP_HEIGHT(plv, pgrp); } } } if (!fPartialOK) rc = rcBounds; // Scrolling is done relative to this calculated rect, not the size of hwndListview (plv->sizeClient) ListView_GetClientRect(plv, &rcClient, TRUE, NULL); ASSERT(ListView_ValidateScrollPositions(plv, &rcClient)); // If any part of rc is outside of rcClient, then // scroll so that all of rcBounds is visible. // dx = 0; if (rc.left < 0 || (rc.right >= rcClient.right && rcClient.right != 0)) { dx = rcBounds.left - 0; if (dx >= 0) { dx = rcBounds.right - rcClient.right; if (dx <= 0) dx = 0; else if ((rcBounds.left - dx) < 0) dx = rcBounds.left - 0; // Not all fits... } } dy = 0; if (rc.top < 0 || (rc.bottom >= rcClient.bottom && rcClient.bottom != 0)) { dy = rcBounds.top - 0; if (dy >= 0) { dy = rcBounds.bottom - rcClient.bottom; if (dy < 0) dy = 0; } } // if rcClient is 0 or 1 pixel in size, it is impossible to scroll it if (dx | dy) ListView_ValidateScrollParams(plv, &dx, &dy); if (dx | dy) return ListView_OnScrollSelectSmooth(plv, dx, dy, SSW_EX_IMMEDIATE); return TRUE; } void ListView_UpdateScrollBars(LV* plv) { RECT rc; DWORD dwStyle; if ((plv->ci.style & LVS_NOSCROLL) || (!(ListView_RedrawEnabled(plv)))) return; _ListView_UpdateScrollBars(plv); GetClientRect(plv->ci.hwnd, &rc); plv->sizeClient.cx = rc.right; plv->sizeClient.cy = rc.bottom; dwStyle = ListView_GetWindowStyle(plv); plv->ci.style = (plv->ci.style & ~(WS_HSCROLL | WS_VSCROLL)) | (dwStyle & WS_HSCROLL | WS_VSCROLL); } #ifndef WINNT #pragma optimize ("gle", off) // Crappy hack for Sage, which passes unitialized memory to SetWindowPlacement. // They used to get lucky and get zeros for the max position, but now they end // up with non-zero stack trash that causes bad things to happen when sage is // maximized. Thus, zero a bunch of stack to save their tail... void ZeroSomeStackForSage() { BYTE aByte[1024]; memset(aByte, 0, sizeof(aByte)); aByte; } #pragma optimize ("", on) #endif void ListView_OnSetFont(LV* plv, HFONT hfont, BOOL fRedraw) { HDC hdc; SIZE siz; LOGFONT lf; HFONT hfontPrev; TEXTMETRIC tm; if ((plv->flags & LVF_FONTCREATED) && plv->hfontLabel) { DeleteObject(plv->hfontLabel); plv->flags &= ~LVF_FONTCREATED; } if (hfont == NULL) { SystemParametersInfo(SPI_GETICONTITLELOGFONT, sizeof(lf), &lf, FALSE); hfont = CreateFontIndirect(&lf); plv->flags |= LVF_FONTCREATED; } hdc = GetDC(HWND_DESKTOP); if (hdc) { hfontPrev = SelectFont(hdc, hfont); GetTextMetrics(hdc, &tm); plv->cyLabelChar = tm.tmHeight; plv->cxLabelChar = tm.tmAveCharWidth; // Maybe this should tm.tmMaxCharWidth GetTextExtentPoint(hdc, c_szEllipses, CCHELLIPSES, &siz); plv->cxEllipses = siz.cx; SelectFont(hdc, hfontPrev); ReleaseDC(HWND_DESKTOP, hdc); } plv->hfontLabel = hfont; if (plv->hfontLabel) { LOGFONT lf; if (GetObject(plv->hfontLabel, sizeof(LOGFONT), &lf)) { if (plv->hfontGroup) DeleteObject(plv->hfontGroup); CCAdjustForBold(&lf); plv->hfontGroup = CreateFontIndirect(&lf); } } plv->ci.uiCodePage = GetCodePageForFont(hfont); ListView_InvalidateCachedLabelSizes(plv); /* Ensure that our tooltip control uses the same font as the list view is using, therefore / avoiding any nasty formatting problems. */ if (plv->hwndToolTips) { FORWARD_WM_SETFONT(plv->hwndToolTips, plv->hfontLabel, FALSE, SendMessage); } // If we have a header window, we need to forward this to it also // as we have destroyed the hfont that they are using... if (plv->hwndHdr) { FORWARD_WM_SETFONT(plv->hwndHdr, plv->hfontLabel, FALSE, SendMessage); ListView_UpdateScrollBars(plv); } if (plv->hFontHot) { DeleteObject(plv->hFontHot); plv->hFontHot = NULL; } CCGetHotFont(plv->hfontLabel, &plv->hFontHot); plv->iFreeSlot = -1; if (fRedraw) RedrawWindow(plv->ci.hwnd, NULL, NULL, RDW_INVALIDATE | RDW_ERASE); #ifndef WINNT ZeroSomeStackForSage(); #endif } HFONT ListView_OnGetFont(LV* plv) { return plv->hfontLabel; } // This function process the WM_SETREDRAW message by setting or clearing // a bit in the listview structure, which several places in the code will // check... // // REVIEW: Should probably forward to DefWindowProc() // void ListView_OnSetRedraw(LV* plv, BOOL fRedraw) { if (fRedraw) { BOOL fChanges = FALSE; // Only do work if we're turning redraw back on... // if (!(plv->flags & LVF_REDRAW)) { plv->flags |= LVF_REDRAW; // deal with any accumulated invalid regions if (plv->hrgnInval) { UINT fRedraw = (plv->flags & LVF_ERASE) ? RDW_INVALIDATE | RDW_ERASE | RDW_UPDATENOW : RDW_UPDATENOW|RDW_INVALIDATE; if (plv->hrgnInval == (HRGN)ENTIRE_REGION) plv->hrgnInval = NULL; RedrawWindow(plv->ci.hwnd, NULL, plv->hrgnInval, fRedraw); ListView_DeleteHrgnInval(plv); fChanges = TRUE; } plv->flags &= ~LVF_ERASE; // Turning redraw on recomputes listview. if (plv->fGroupView) { _ListView_RecomputeEx(plv, NULL, 0, TRUE); } if (plv->fGroupView || ListView_IsDoubleBuffer(plv)) InvalidateRect(plv->ci.hwnd, NULL, TRUE); // now deal with the optimized stuff if (ListView_IsListView(plv) || ListView_IsReportView(plv)) { if (plv->iFirstChangedNoRedraw != -1) { // We may try to resize the column if (!ListView_MaybeResizeListColumns(plv, plv->iFirstChangedNoRedraw, ListView_Count(plv)-1)) ListView_OnUpdate(plv, plv->iFirstChangedNoRedraw); } else ListView_UpdateScrollBars(plv); } else { int iCount; if (plv->iFirstChangedNoRedraw != -1) { for (iCount = ListView_Count(plv) ; plv->iFirstChangedNoRedraw < iCount; plv->iFirstChangedNoRedraw++) { ListView_InvalidateItem(plv, plv->iFirstChangedNoRedraw, FALSE, RDW_INVALIDATE); } fChanges = TRUE; } if (fChanges) { ListView_RecalcRegion(plv, TRUE, TRUE); } if (((plv->ci.style & LVS_AUTOARRANGE) ||(plv->exStyle & LVS_EX_SNAPTOGRID)) && fChanges) { ListView_OnUpdate(plv, plv->iFirstChangedNoRedraw); } else { ListView_UpdateScrollBars(plv); } } } } else { plv->iFirstChangedNoRedraw = -1; plv->flags &= ~LVF_REDRAW; } } HIMAGELIST ListView_OnGetImageList(LV* plv, int iImageList) { switch (iImageList) { case LVSIL_NORMAL: return plv->himl; case LVSIL_SMALL: return plv->himlSmall; case LVSIL_STATE: return plv->himlState; } RIPMSG(0, "ListView_GetImageList: Invalid Imagelist asked for"); return NULL; } HIMAGELIST ListView_OnSetImageList(LV* plv, HIMAGELIST himl, int iImageList) { HIMAGELIST hImageOld = NULL; BOOL fImageSizeChanged = FALSE; //Assume the size hasn't changed! switch (iImageList) { case LVSIL_NORMAL: hImageOld = plv->himl; plv->himl = himl; if (himl) { int cxIconNew, cyIconNew; //Get the Icon sizes from the new image list. if (CCGetIconSize(&plv->ci, himl, &cxIconNew , &cyIconNew)) { //Check to see if the sizes have changed! if((cxIconNew != plv->cxIcon) || (cyIconNew != plv->cyIcon)) { fImageSizeChanged = TRUE; plv->cxIcon = cxIconNew; plv->cyIcon = cyIconNew; } } if (fImageSizeChanged && (!(plv->flags & LVF_ICONSPACESET))) { ListView_OnSetIconSpacing(plv, (LPARAM)-1); } } break; case LVSIL_SMALL: hImageOld = plv->himlSmall; plv->himlSmall = himl; if (himl) { int cxSmIconNew, cySmIconNew; //Get the small icon sizes from the new image list. if(CCGetIconSize(&plv->ci, himl, &cxSmIconNew , &cySmIconNew)) { //Check to see if the sizes have changed! if((cxSmIconNew != plv->cxSmIcon) || (cySmIconNew != plv->cySmIcon)) { fImageSizeChanged = TRUE; plv->cxSmIcon = cxSmIconNew; plv->cySmIcon = cySmIconNew; } } } if (fImageSizeChanged) { plv->cxItem = ListView_ComputeCXItemSize(plv); // After changing the imagelist, try to resize the columns, because we can't // guess what the new size is going to be. Discovered by Thumbview ListView_MaybeResizeListColumns(plv, 0, ListView_Count(plv)-1); plv->cyItem = ListView_ComputeCYItemSize(plv); } if (plv->hwndHdr) SendMessage(plv->hwndHdr, HDM_SETIMAGELIST, 0, (LPARAM)himl); break; case LVSIL_STATE: fImageSizeChanged = TRUE; if (himl) { CCGetIconSize(&plv->ci, himl, &plv->cxState , &plv->cyState); } else { plv->cxState = 0; } hImageOld = plv->himlState; plv->himlState = himl; plv->cyItem = ListView_ComputeCYItemSize(plv); break; default: fImageSizeChanged = TRUE; TraceMsg(TF_LISTVIEW, "sh TR - LVM_SETIMAGELIST: unrecognized iImageList"); break; } if (himl && !(plv->ci.style & LVS_SHAREIMAGELISTS)) ImageList_SetBkColor(himl, plv->clrBk); // Imagelist size changed... if we're in tileview, we need to recalculate the tilesize. if (ListView_IsTileView(plv) && (iImageList == LVSIL_STATE || iImageList == LVSIL_NORMAL)) { ListView_RecalcTileSize(plv); } if(fImageSizeChanged) { // Now, recompute! plv->rcView.left = RECOMPUTE; // invalidate this up front to avoid the asserts - it'll get recalculated anyway _ListView_RecomputeEx(plv, NULL, 0, TRUE); } if (ListView_Count(plv) > 0) InvalidateRect(plv->ci.hwnd, NULL, TRUE); return hImageOld; } #ifdef UNICODE BOOL ListView_OnGetItemA(LV* plv, LV_ITEMA *plvi) { LPWSTR pszW = NULL; LPSTR pszC = NULL; BOOL fRet; //HACK ALERT -- this code assumes that LV_ITEMA is exactly the same // as LV_ITEMW except for the pointer to the string. COMPILETIME_ASSERT(sizeof(LV_ITEMA) == sizeof(LV_ITEMW)) if (!plvi) return FALSE; if ((plvi->mask & LVIF_TEXT) && (plvi->pszText != NULL)) { pszC = plvi->pszText; pszW = LocalAlloc(LMEM_FIXED, plvi->cchTextMax * sizeof(WCHAR)); if (pszW == NULL) return FALSE; plvi->pszText = (LPSTR)pszW; } fRet = ListView_OnGetItem(plv, (LV_ITEM *) plvi); if (pszW) { if (plvi->pszText != LPSTR_TEXTCALLBACKA) { if (fRet && plvi->cchTextMax) ConvertWToAN(plv->ci.uiCodePage, pszC, plvi->cchTextMax, (LPWSTR)plvi->pszText, -1); plvi->pszText = pszC; } LocalFree(pszW); } return fRet; } #endif BOOL ListView_OnGetItem(LV* plv, LV_ITEM* plvi) { UINT mask; LISTITEM* pitem = NULL; LV_DISPINFO nm; if (!plvi) { RIPMSG(0, "LVM_GET(ITEM|ITEMTEXT): Invalid pitem = NULL"); return FALSE; } if (!ListView_IsValidItemNumber(plv, plvi->iItem)) { #ifdef DEBUG // owner data views (e.g. docfind) may change the number of items in listview // while we are doing something, thus hitting this rip if (!ListView_IsOwnerData(plv)) RIPMSG(0, "LVM_GET(ITEM|ITEMTEXT|ITEMSTATE): item=%d does not exist", plvi->iItem); #endif return FALSE; } nm.item.mask = 0; mask = plvi->mask; if (!ListView_IsOwnerData(plv)) { // Standard listviews pitem = ListView_FastGetItemPtr(plv, plvi->iItem); ASSERT(pitem); // Handle sub-item cases for report view // if (plvi->iSubItem != 0) { LISTSUBITEM lsi; ListView_GetSubItem(plv, plvi->iItem, plvi->iSubItem, &lsi); if (mask & LVIF_TEXT) { if (lsi.pszText != LPSTR_TEXTCALLBACK) { Str_GetPtr0(lsi.pszText, plvi->pszText, plvi->cchTextMax); } else { // if this is LVIF_NORECOMPUTE we will update pszText later nm.item.mask |= LVIF_TEXT; } } if ((mask & LVIF_IMAGE) && (plv->exStyle & LVS_EX_SUBITEMIMAGES)) { plvi->iImage = lsi.iImage; if (lsi.iImage == I_IMAGECALLBACK) nm.item.mask |= LVIF_IMAGE; } if (mask & LVIF_STATE) { if (ListView_FullRowSelect(plv)) { // if we're in full row select, // the state bit for select and focus follows column 0. lsi.state |= pitem->state & (LVIS_SELECTED | LVIS_FOCUSED | LVIS_DROPHILITED); } plvi->state = lsi.state & plvi->stateMask; if (plv->stateCallbackMask) { nm.item.stateMask = (plvi->stateMask & plv->stateCallbackMask); if (nm.item.stateMask) { nm.item.mask |= LVIF_STATE; nm.item.state = 0; } } } } else { if (mask & LVIF_TEXT) { if (pitem->pszText != LPSTR_TEXTCALLBACK) { Str_GetPtr0(pitem->pszText, plvi->pszText, plvi->cchTextMax); } else { // if this is LVIF_NORECOMPUTE we will update pszText later nm.item.mask |= LVIF_TEXT; } } if (mask & LVIF_IMAGE) { plvi->iImage = pitem->iImage; if (pitem->iImage == I_IMAGECALLBACK) nm.item.mask |= LVIF_IMAGE; } if (mask & LVIF_INDENT) { plvi->iIndent = pitem->iIndent; if (pitem->iIndent == I_INDENTCALLBACK) nm.item.mask |= LVIF_INDENT; } if (mask & LVIF_STATE) { plvi->state = (pitem->state & plvi->stateMask); if (plv->stateCallbackMask) { nm.item.stateMask = (plvi->stateMask & plv->stateCallbackMask); if (nm.item.stateMask) { nm.item.mask |= LVIF_STATE; nm.item.state = 0; } } } if (mask & LVIF_GROUPID) { if (LISTITEM_HASGROUP(pitem)) { plvi->iGroupId = pitem->pGroup->iGroupId; } else { nm.item.mask |= LVIF_GROUPID; } } if (mask & LVIF_COLUMNS) { if ((plvi->puColumns == NULL) || (plvi->cColumns > CCMAX_TILE_COLUMNS)) { return FALSE; } if (pitem->cColumns == I_COLUMNSCALLBACK) { nm.item.mask |= LVIF_COLUMNS; } else { plvi->cColumns = pitem->cColumns; if (plvi->cColumns < pitem->cColumns) { // Not enough room to store the columns return FALSE; } // Copy the array if (plvi->puColumns && pitem->puColumns) { CopyMemory(plvi->puColumns, pitem->puColumns, plvi->cColumns * sizeof(UINT)); } } } } if (mask & LVIF_PARAM) plvi->lParam = pitem->lParam; } else { // Complete call back for info... // Handle sub-item cases for report view // if (plvi->iSubItem != 0) { // if there are no subitem images, don't query for them if (!(plv->exStyle & LVS_EX_SUBITEMIMAGES)) mask &= ~LVIF_IMAGE; // don't allow indent on the non-0th column mask &= ~LVIF_INDENT; } if (mask & LVIF_PARAM) plvi->lParam = 0L; // Dont have any to return now... if (mask & LVIF_STATE) { plvi->state = 0; if ((plvi->iSubItem == 0) || ListView_FullRowSelect(plv)) { if (plvi->iItem == plv->iFocus) plvi->state |= LVIS_FOCUSED; if (plv->plvrangeSel->lpVtbl->IsSelected(plv->plvrangeSel, plvi->iItem) == S_OK) plvi->state |= LVIS_SELECTED; if (plv->plvrangeCut->lpVtbl->IsSelected(plv->plvrangeCut, plvi->iItem) == S_OK) plvi->state |= LVIS_CUT; if (plvi->iItem == plv->iDropHilite) plvi->state |= LVIS_DROPHILITED; plvi->state &= plvi->stateMask; } if (plv->stateCallbackMask) { nm.item.stateMask = (plvi->stateMask & plv->stateCallbackMask); if (nm.item.stateMask) { nm.item.mask |= LVIF_STATE; nm.item.state = 0; } } } if (mask & LVIF_COLUMNS) { nm.item.mask |= LVIF_COLUMNS; } nm.item.mask |= (mask & (LVIF_TEXT | LVIF_IMAGE | LVIF_INDENT)); } if (mask & LVIF_NORECOMPUTE) { if (nm.item.mask & LVIF_TEXT) plvi->pszText = LPSTR_TEXTCALLBACK; if (nm.item.mask & LVIF_COLUMNS) plvi->cColumns = I_COLUMNSCALLBACK; } else if (nm.item.mask) { UINT rguColumns[CCMAX_TILE_COLUMNS]; nm.item.iItem = plvi->iItem; nm.item.iSubItem = plvi->iSubItem; if (ListView_IsOwnerData(plv)) nm.item.lParam = 0L; else nm.item.lParam = pitem->lParam; // just in case LVIF_IMAGE is set and callback doesn't fill it in // ... we'd rather have a -1 than whatever garbage is on the stack nm.item.iImage = -1; nm.item.iIndent = 0; if (nm.item.mask & LVIF_TEXT) { RIPMSG(plvi->pszText != NULL, "LVM_GET(ITEM|ITEMTEXT) null string pointer"); if (plvi->pszText) { nm.item.pszText = plvi->pszText; nm.item.cchTextMax = plvi->cchTextMax; // Make sure the buffer is zero terminated... if (nm.item.cchTextMax) *nm.item.pszText = 0; } else { // Don't make caller smash null pointer nm.item.mask &= ~LVIF_TEXT; } } if (nm.item.mask & LVIF_COLUMNS) { nm.item.cColumns = plvi->cColumns; nm.item.puColumns = rguColumns; if (plvi->puColumns && plvi->cColumns && plvi->cColumns < ARRAYSIZE(rguColumns)) { CopyMemory(rguColumns, plvi->puColumns, sizeof(UINT) * plvi->cColumns); } } CCSendNotify(&plv->ci, LVN_GETDISPINFO, &nm.hdr); // use nm.item.mask to give the app a chance to change values if (nm.item.mask & LVIF_INDENT) plvi->iIndent = nm.item.iIndent; if (nm.item.mask & LVIF_GROUPID) { if (pitem) { if (nm.item.iGroupId == I_GROUPIDNONE) { ListView_RemoveItemFromItsGroup(plv, pitem); LISTITEM_SETASKEDFORGROUP(pitem); } else { LISTGROUP* pgrp = ListView_FindGroupFromID(plv, nm.item.iGroupId, NULL); if (pgrp != pitem->pGroup) { ListView_RemoveItemFromItsGroup(plv, pitem); pitem->pGroup = pgrp; if (pgrp) { DPA_AppendPtr(pgrp->hdpa, pitem); } } } } plvi->iGroupId = nm.item.iGroupId; } if (nm.item.mask & LVIF_STATE) plvi->state ^= ((plvi->state ^ nm.item.state) & nm.item.stateMask); if (nm.item.mask & LVIF_IMAGE) plvi->iImage = nm.item.iImage; if (nm.item.mask & LVIF_TEXT) plvi->pszText = CCReturnDispInfoText(nm.item.pszText, plvi->pszText, plvi->cchTextMax); if (nm.item.mask & LVIF_COLUMNS) { // Put the # of columns back in the LV_ITEM struct. Don't need to // do anything with puColumns. UINT cColumns = (nm.item.cColumns == I_COLUMNSCALLBACK) ? 0 : nm.item.cColumns; UINT cColumnsToCopy = min(cColumns, plvi->cColumns); // Copy rguColumns back into the thing we were passed. CopyMemory(plvi->puColumns, rguColumns, sizeof(UINT) * cColumnsToCopy); plvi->cColumns = cColumnsToCopy; } if (pitem && (nm.item.mask & LVIF_DI_SETITEM)) { // // The SendNotify above can set about a terrible series of events // whereby asking for DISPINFO causes the shell to look around // (call peekmessage) to see if its got a new async icon for the // listview. This lets other messages be delivered, such as an // UPDATEIMAGE of Index == -1 (if the user is changing icon sizing // at the same time). This causes a re-enumeration of the desktop // and hence this very listview is torn down and rebuilt while // we're sitting here for the DISPINFO to finish. Thus, as a cheap // and dirty solution, I check to see if the item I think I have // is the same one I had when I made the notify, and if not, I // bail. Don't blame me, I'm just cleaning up the mess. if (pitem != ListView_GetItemPtr(plv, plvi->iItem)) { return FALSE; } if (nm.item.iSubItem == 0) { //DebugMsg(TF_LISTVIEW, "SAVING ITEMS!"); if (nm.item.mask & LVIF_IMAGE) pitem->iImage = (short) nm.item.iImage; if (nm.item.mask & LVIF_INDENT) pitem->iIndent = (short) nm.item.iIndent; if (nm.item.mask & LVIF_TEXT) if (nm.item.pszText) { Str_Set(&pitem->pszText, nm.item.pszText); } if (nm.item.mask & LVIF_STATE) pitem->state ^= ((pitem->state ^ nm.item.state) & nm.item.stateMask); if (nm.item.mask & LVIF_COLUMNS) { Tile_Set(&pitem->puColumns, &pitem->cColumns, nm.item.puColumns, nm.item.cColumns); // Just did a Tile_Set - need to recompute. ListView_SetSRecompute(pitem); } } else { ListView_SetSubItem(plv, &nm.item); } } } return TRUE; } BOOL ListView_OnSetItemA(LV* plv, LV_ITEMA* plvi) { LPWSTR pszW = NULL; LPSTR pszC = NULL; BOOL fRet; // Let ListView_OnSetItem() handle owner-data validation //HACK ALERT -- this code assumes that LV_ITEMA is exactly the same // as LV_ITEMW except for the pointer to the string. COMPILETIME_ASSERT(sizeof(LV_ITEMA) == sizeof(LV_ITEMW)); if (!plvi) return FALSE; if ((plvi->mask & LVIF_TEXT) && (plvi->pszText != NULL)) { pszC = plvi->pszText; pszW = ProduceWFromA(plv->ci.uiCodePage, pszC); if (pszW == NULL) return FALSE; plvi->pszText = (LPSTR)pszW; } fRet = ListView_OnSetItem(plv, (const LV_ITEM*) plvi); if (pszW != NULL) { plvi->pszText = pszC; FreeProducedString(pszW); } return fRet; } void ListView_OffsetRect(LV* plv, RECT* prc) { if (ListView_IsReportView(plv)) { OffsetRect(prc, -plv->ptlRptOrigin.x, -plv->ptlRptOrigin.y + plv->yTop); } else { OffsetRect(prc, -plv->ptOrigin.x, -plv->ptOrigin.y); } } BOOL ListView_OnSetItem(LV* plv, const LV_ITEM* plvi) { LISTITEM* pitem = NULL; UINT mask; UINT maskChanged; UINT rdwFlags=RDW_INVALIDATE; int i; UINT stateOld, stateNew; BOOL fFocused = FALSE; BOOL fSelected = FALSE; BOOL fStateImageChanged = FALSE; if (ListView_IsOwnerData(plv)) { RIPMSG(0, "LVM_SETITEM: Invalid for owner-data listview"); return FALSE; } if (!plvi) return FALSE; RIPMSG(plvi->iSubItem >= 0, "ListView_OnSetItem: Invalid item index"); if (plv->himl && (plv->clrBk != ImageList_GetBkColor(plv->himl))) rdwFlags |= RDW_ERASE; mask = plvi->mask; if (!mask) return TRUE; // If we're setting a subitem, handle it elsewhere... // if (plvi->iSubItem > 0) return ListView_SetSubItem(plv, plvi); i = plvi->iItem; pitem = ListView_GetItemPtr(plv, i); if (!pitem) return FALSE; //REVIEW: This is a BOGUS HACK, and should be fixed. //This incorrectly calculates the old state (since we may // have to send LVN_GETDISPINFO to get it). // stateOld = stateNew = 0; if (mask & LVIF_STATE) { stateOld = pitem->state & plvi->stateMask; stateNew = plvi->state & plvi->stateMask; } // Prevent multiple selections in a single-select listview. if ((plv->ci.style & LVS_SINGLESEL) && (mask & LVIF_STATE) && (stateNew & LVIS_SELECTED)) { ListView_DeselectAll(plv, i); // Refresh the old state information stateOld = pitem->state & plvi->stateMask; } if (!ListView_SendChange(plv, i, 0, LVN_ITEMCHANGING, stateOld, stateNew, mask, pitem->lParam)) return FALSE; maskChanged = 0; if (mask & LVIF_STATE) { UINT change = (pitem->state ^ plvi->state) & plvi->stateMask; if (change) { pitem->state ^= change; maskChanged |= LVIF_STATE; // the selection state has changed.. update selected count if (change & LVIS_SELECTED) { fSelected = TRUE; if (pitem->state & LVIS_SELECTED) { plv->nSelected++; } else { if (plv->nSelected > 0) plv->nSelected--; } } // For some bits we can only invert the label area... // fSelectOnlyChange = ((change & ~(LVIS_SELECTED | LVIS_FOCUSED | LVIS_DROPHILITED)) == 0); // fEraseItem = ((change & ~(LVIS_SELECTED | LVIS_DROPHILITED)) != 0); // try to steal focus from the previous guy. if (change & LVIS_FOCUSED) { BOOL fUnfolded = ListView_IsItemUnfolded(plv, plv->iFocus); int iOldFocus = plv->iFocus; RECT rcLabel; fFocused = TRUE; if (plv->iFocus != i) { if ((plv->iFocus == -1) || ListView_OnSetItemState(plv, plv->iFocus, 0, LVIS_FOCUSED)) { ASSERT(pitem->state & LVIS_FOCUSED); plv->iFocus = i; if (plv->iMark == -1) plv->iMark = i; } else { fFocused = FALSE; pitem->state &= ~LVIS_FOCUSED; } } else { ASSERT(!(pitem->state & LVIS_FOCUSED)); plv->iFocus = -1; } // If we were previously unfolded and we move the focus we must // attempt to refresh the previous focus owner to referect this change. if (fUnfolded && !ListView_IsItemUnfolded(plv, iOldFocus) && (plv->iItemDrawing != iOldFocus)) { ListView_GetUnfoldedRect(plv, iOldFocus, &rcLabel); RedrawWindow(plv->ci.hwnd, &rcLabel, NULL, RDW_INVALIDATE|RDW_ERASE); } // Kill the tooltip if focus moves, it causes us headaches otherwise! ListView_PopBubble(plv); } if (change & LVIS_CUT || plv->clrTextBk == CLR_NONE) rdwFlags |= RDW_ERASE; if (change & LVIS_OVERLAYMASK) { // Overlay changed, so need to blow away icon region cache if (pitem->hrgnIcon) { if (pitem->hrgnIcon != (HANDLE) -1) DeleteObject(pitem->hrgnIcon); pitem->hrgnIcon = NULL; } } fStateImageChanged = (change & LVIS_STATEIMAGEMASK); } } if (mask & LVIF_TEXT) { // need to do this now because we're changing the text // so we need to get the rect of the thing before the text changes // but don't redraw the item we are currently painting if (plv->iItemDrawing != i) { ListView_InvalidateItemEx(plv, i, FALSE, RDW_INVALIDATE | RDW_ERASE, LVIF_TEXT); } if (!Str_Set(&pitem->pszText, plvi->pszText)) return FALSE; plv->rcView.left = RECOMPUTE; ListView_SetSRecompute(pitem); maskChanged |= LVIF_TEXT; } if (mask & LVIF_INDENT) { if (pitem->iIndent != plvi->iIndent) { pitem->iIndent = (short) plvi->iIndent; maskChanged |= LVIF_INDENT; if (ListView_IsReportView(plv)) rdwFlags |= RDW_ERASE; } } if (mask & LVIF_IMAGE) { if (pitem->iImage != plvi->iImage) { pitem->iImage = (short) plvi->iImage; maskChanged |= LVIF_IMAGE; if (pitem->hrgnIcon) { if (pitem->hrgnIcon != (HANDLE) -1) DeleteObject(pitem->hrgnIcon); pitem->hrgnIcon = NULL; } // erase if there was a set image if (pitem->iImage != I_IMAGECALLBACK) rdwFlags |= RDW_ERASE; } } if (mask & LVIF_PARAM) { if (pitem->lParam != plvi->lParam) { pitem->lParam = plvi->lParam; maskChanged |= LVIF_PARAM; } } if (mask & LVIF_GROUPID) { LISTGROUP* pgrp = ListView_FindGroupFromID(plv, plvi->iGroupId, NULL); if (pgrp) { if (pitem->pGroup != pgrp) { ListView_RemoveItemFromItsGroup(plv, pitem); pitem->pGroup = pgrp; DPA_AppendPtr(pgrp->hdpa, pitem); if (ListView_RedrawEnabled(plv)) { _ListView_RecomputeEx(plv, NULL, 0, FALSE); InvalidateRect(plv->ci.hwnd, NULL, TRUE); } maskChanged |= LVIF_GROUPID; } } } if (mask & LVIF_COLUMNS) { UINT uNumColumns = (plvi->cColumns == I_COLUMNSCALLBACK) ? 0 : plvi->cColumns; if (((uNumColumns > 0) && (plvi->puColumns == NULL)) || // Didn't provide any columns (uNumColumns > CCMAX_TILE_COLUMNS)) // Provided too many { return FALSE; // See note below about premature return. } if (!Tile_Set(&pitem->puColumns, &pitem->cColumns, plvi->puColumns, plvi->cColumns)) return FALSE; // Note: if we fail here, we may have still set the LVIF_TEXT above... // so the call partially succeeded. Oh well, no way to undo that. maskChanged |= LVIF_COLUMNS; // Columns changed - need to recompute this guy. ListView_SetSRecompute(pitem); } if (maskChanged) { // don't redraw the item we are currently painting if (plv->iItemDrawing != i) ListView_InvalidateItemEx(plv, i, FALSE, rdwFlags, maskChanged); TraceMsg(DM_LVSENDCHANGE, "LV - SendChange %d %d %d %d", i, stateOld, stateNew, maskChanged); ListView_SendChange(plv, i, 0, LVN_ITEMCHANGED, stateOld, stateNew, maskChanged, pitem->lParam); if (maskChanged & LVIF_TEXT) NotifyWinEvent(EVENT_OBJECT_NAMECHANGE, plv->ci.hwnd, OBJID_CLIENT, i+1); if (maskChanged & LVIF_STATE) { if (fFocused) ListView_NotifyFocusEvent(plv); if (fSelected) { if (stateNew & LVIS_SELECTED) { NotifyWinEvent((plv->nSelected == 1) ? EVENT_OBJECT_SELECTION : EVENT_OBJECT_SELECTIONADD, plv->ci.hwnd, OBJID_CLIENT, i+1); } else { NotifyWinEvent(EVENT_OBJECT_SELECTIONREMOVE, plv->ci.hwnd, OBJID_CLIENT, i+1); } } if (fStateImageChanged) NotifyWinEvent(EVENT_OBJECT_STATECHANGE, plv->ci.hwnd, OBJID_CLIENT, i+1); } } return TRUE; } UINT ListView_OnGetItemState(LV* plv, int i, UINT mask) { LV_ITEM lvi; lvi.mask = LVIF_STATE; lvi.stateMask = mask; lvi.iItem = i; lvi.iSubItem = 0; if (!ListView_OnGetItem(plv, &lvi)) return 0; return lvi.state; } BOOL ListView_OnSetItemState(LV* plv, int i, UINT data, UINT mask) { UINT rdwFlags = RDW_INVALIDATE; LV_ITEM lvi; lvi.mask = LVIF_STATE; lvi.state = data; lvi.stateMask = mask; lvi.iItem = i; lvi.iSubItem = 0; // if the item is -1, we will do it for all items. We special case // a few cases here as to speed it up. For example if the mask is // LVIS_SELECTED and data is zero it implies that we will deselect // all items... // if (ListView_IsOwnerData(plv)) { UINT uOldData = 0; // these are the only two we handled mask &= (LVIS_SELECTED | LVIS_FOCUSED | LVIS_CUT | LVIS_DROPHILITED); if (!mask) return TRUE; if (plv->clrTextBk == CLR_NONE || (plv->himl && (plv->clrBk != ImageList_GetBkColor(plv->himl)))) { rdwFlags |= RDW_ERASE; } if (i == -1) { // request selection state change for all if (mask & LVIS_SELECTED) { if (data & LVIS_SELECTED) { // set selection if ((plv->ci.style & LVS_SINGLESEL)) { // cant make multiple selections in a single-select listview. return FALSE; } if (plv->cTotalItems) { if (FAILED(plv->plvrangeSel->lpVtbl->IncludeRange(plv->plvrangeSel, 0, plv->cTotalItems - 1))) return FALSE; } RedrawWindow(plv->ci.hwnd, NULL, NULL, rdwFlags); } else { // clear selection if (plv->nSelected > 0) { ListView_InvalidateSelectedOrCutOwnerData(plv, plv->plvrangeSel); if (FAILED(plv->plvrangeSel->lpVtbl->Clear(plv->plvrangeSel))) return FALSE; } else { // if nothing was selected, then there's nothing to clear // no change. mask &= ~ LVIS_SELECTED; } } uOldData |= (LVIS_SELECTED & (mask ^ data)); // Update our internal count to what the list thinks is the number selected... plv->plvrangeSel->lpVtbl->CountIncluded(plv->plvrangeSel, &plv->nSelected); } // can maybe combine with above code... if (mask & LVIS_CUT) { if (data & LVIS_CUT) { // set selection if (plv->cTotalItems) if (FAILED(plv->plvrangeCut->lpVtbl->IncludeRange(plv->plvrangeCut, 0, plv->cTotalItems - 1))) return FALSE; RedrawWindow(plv->ci.hwnd, NULL, NULL, rdwFlags); } else { // clear selection if (plv->plvrangeCut->lpVtbl->IsEmpty(plv->plvrangeCut) != S_OK) { ListView_InvalidateSelectedOrCutOwnerData(plv, plv->plvrangeCut); if (FAILED(plv->plvrangeCut->lpVtbl->Clear(plv->plvrangeCut))) return FALSE; } else { // if nothing was selected, then there's nothing to clear // no change. mask &= ~ LVIS_CUT; } } uOldData |= (LVIS_CUT & (mask ^ data)); } // request focus state change if (mask & LVIS_FOCUSED) { if (data & LVIS_FOCUSED) { // cant set focus to all return FALSE; } else if (plv->iFocus != -1) { int iOldFocus = plv->iFocus; // clear focus uOldData |= (LVIS_FOCUSED & (mask ^ data)); plv->iFocus = -1; // notify that the old focus is being lost DebugMsg(DM_LVSENDCHANGE, TEXT("VLV: LVN_ITEMCHANGED: %d %d %d"), iOldFocus, LVIS_FOCUSED, 0); ListView_SendChange(plv, iOldFocus, 0, LVN_ITEMCHANGED, LVIS_FOCUSED, 0, LVIF_STATE, 0); ListView_InvalidateFoldedItem(plv, iOldFocus, TRUE, RDW_INVALIDATE |RDW_ERASE); } } if (mask & LVIS_DROPHILITED) { if (data & LVIS_DROPHILITED) { // cant set focus to all return FALSE; } else if (plv->iDropHilite != -1) { int iOldDropHilite = plv->iDropHilite; // clear focus uOldData |= (LVIS_FOCUSED & (mask ^ data)); plv->iDropHilite = -1; // notify that the old focus is being lost ListView_SendChange(plv, iOldDropHilite, 0, LVN_ITEMCHANGED, LVIS_DROPHILITED, 0, LVIF_STATE, 0); ListView_InvalidateFoldedItem(plv, iOldDropHilite, TRUE, RDW_INVALIDATE |RDW_ERASE); } } // invalidate and notify if there was a change if (uOldData ^ (data & mask)) { DebugMsg(DM_LVSENDCHANGE, TEXT("VLV: LVN_ITEMCHANGED: %d %d %d"), i, uOldData, data); ListView_SendChange(plv, i, 0, LVN_ITEMCHANGED, uOldData, data, LVIF_STATE, 0); if (mask & LVIS_SELECTED) { // Tell accessibility, "Selection changed in a complex way" // (There is no "select all" or "select none" notification) NotifyWinEvent(EVENT_OBJECT_SELECTIONWITHIN, plv->ci.hwnd, OBJID_CLIENT, CHILDID_SELF); } } } else { if (!ListView_IsValidItemNumber(plv, i)) return FALSE; // request selection state change // and the selection state is new... if ((mask & LVIS_SELECTED)) { if (((plv->plvrangeSel->lpVtbl->IsSelected(plv->plvrangeSel, i) == S_OK) ? LVIS_SELECTED : 0) ^ (data & LVIS_SELECTED)) { if (data & LVIS_SELECTED) { // set selection if ((plv->ci.style & LVS_SINGLESEL)) { // in single selection mode, we need to deselect everything else if (!ListView_OnSetItemState(plv, -1, 0, LVIS_SELECTED)) return FALSE; } // now select the new item if (FAILED(plv->plvrangeSel->lpVtbl->IncludeRange(plv->plvrangeSel, i, i))) return FALSE; } else { // clear selection if (FAILED(plv->plvrangeSel->lpVtbl->ExcludeRange(plv->plvrangeSel, i, i))) return FALSE; } // something actually changed (or else we wouldn't be in this // if block uOldData |= (LVIS_SELECTED & (mask ^ data)); } else { // nothing changed... so make the uOldData be the same for this bit // else make this the same as uOldData |= (LVIS_SELECTED & (mask & data)); } // Update our internal count to what the list thinks is the number selected... plv->plvrangeSel->lpVtbl->CountIncluded(plv->plvrangeSel, &plv->nSelected); } if ((mask & LVIS_CUT)) { if (((plv->plvrangeCut->lpVtbl->IsSelected(plv->plvrangeCut, i) == S_OK) ? LVIS_CUT : 0) ^ (data & LVIS_CUT)) { if (data & LVIS_CUT) { // now select the new item if (FAILED(plv->plvrangeCut->lpVtbl->IncludeRange(plv->plvrangeCut, i, i))) return FALSE; } else { // clear selection if (FAILED(plv->plvrangeCut->lpVtbl->ExcludeRange(plv->plvrangeCut, i, i))) return FALSE; } // something actually changed (or else we wouldn't be in this // if block uOldData |= (LVIS_CUT & (mask ^ data)); rdwFlags |= RDW_ERASE; } else { // nothing changed... so make the uOldData be the same for this bit // else make this the same as uOldData |= (LVIS_CUT & (mask & data)); } } // request focus state change if (mask & LVIS_FOCUSED) { int iOldFocus = plv->iFocus; if (data & LVIS_FOCUSED) { // set focus if (i != plv->iFocus) { // we didn't have the focus before plv->iFocus = i; if (plv->iMark == -1) plv->iMark = i; if (iOldFocus != -1) { // we're stealing it from someone // notify of the change DebugMsg(DM_LVSENDCHANGE, TEXT("VLV: LVN_ITEMCHANGED: %d %d %d"), iOldFocus, LVIS_FOCUSED, 0); ListView_SendChange(plv, iOldFocus, 0, LVN_ITEMCHANGED, LVIS_FOCUSED, 0, LVIF_STATE, 0); } } else { // we DID have the focus before uOldData |= LVIS_FOCUSED; } } else { // clear focus if (i == plv->iFocus) { plv->iFocus = -1; uOldData |= LVIS_FOCUSED; } } } // request focus state change if (mask & LVIS_DROPHILITED) { int iOldDropHilite = plv->iDropHilite; if (data & LVIS_DROPHILITED) { // set Drop Hilite if (i != plv->iDropHilite) { // we didn't have the Drop Hilite before plv->iDropHilite = i; if (iOldDropHilite != -1) { // we're stealing it from someone // notify of the change ListView_SendChange(plv, iOldDropHilite, 0, LVN_ITEMCHANGED, LVIS_DROPHILITED, 0, LVIF_STATE, 0); ListView_InvalidateFoldedItem(plv, iOldDropHilite, TRUE, RDW_INVALIDATE |RDW_ERASE); } } else { // we DID have the Drop Hilite before uOldData |= LVIS_DROPHILITED; } } else { // clear Drop Hilite if (i == plv->iDropHilite) { plv->iDropHilite = -1; uOldData |= LVIS_DROPHILITED; } } } // invalidate and notify if there was a change if (uOldData ^ (data & mask)) { DebugMsg(DM_LVSENDCHANGE, TEXT("VLV: LVN_ITEMCHANGED: %d %d %d"), i, uOldData, data); ListView_SendChange(plv, i, 0, LVN_ITEMCHANGED, uOldData, data, LVIF_STATE, 0); ListView_InvalidateItem(plv, i, TRUE, rdwFlags); // Kill the tooltip if focus moves, it causes us headaches otherwise! if ((uOldData ^ (data & mask)) & LVIS_FOCUSED) { ListView_PopBubble(plv); ListView_NotifyFocusEvent(plv); } // Tell accessibility about the changes if (mask & LVIS_SELECTED) { UINT event; if (data & LVIS_SELECTED) { if (plv->nSelected == 1) event = EVENT_OBJECT_SELECTION; // this object is the entire selection else event = EVENT_OBJECT_SELECTIONADD; // this object is selected } else event = EVENT_OBJECT_SELECTIONREMOVE; // this object is unselected NotifyWinEvent(event, plv->ci.hwnd, OBJID_CLIENT, i + 1); } } } } else { if (i != -1) { return ListView_OnSetItem(plv, &lvi); } else { UINT flags = LVNI_ALL; if (data == 0) { switch (mask) { case LVIS_SELECTED: flags = LVNI_SELECTED; break; case LVIS_CUT: flags = LVNI_CUT; break; } } else if ((plv->ci.style & LVS_SINGLESEL) && (mask == LVIS_SELECTED)) return FALSE; /* can't select all in single-select listview */ else if ((mask & data) & LVIS_FOCUSED) { return FALSE; // can't set focus to everything } // // Now iterate over all of the items that match our criteria and // set their new value. // while ((lvi.iItem = ListView_OnGetNextItem(plv, lvi.iItem, flags)) != -1) { ListView_OnSetItem(plv, &lvi); } } } return TRUE; } // // Returns TRUE if the label of an item is not truncated (is unfolded) and FALSE // otherwise. If FALSE is returned, it also fills the Unfolding text in pszText. // If TRUE is returned, pszText is set to empty string. // BOOL ListView_IsItemUnfolded2(LV* plv, int iItem, int iSubItem, LPTSTR pszText, int cchTextMax) { BOOL bItemUnfolded = ListView_IsItemUnfolded(plv, iItem); if (pszText && cchTextMax > 0) // Sanity checks on input params. { pszText[0] = 0; if (!bItemUnfolded) { RECT rcLabel = {0}; LV_ITEM item; item.iItem = iItem; item.iSubItem = iSubItem; item.mask = LVIF_TEXT | LVIF_PARAM; if (ListView_IsTileView(plv)) { TCalculateSubItemRect(plv, NULL, NULL, iItem, iSubItem, NULL, NULL, &bItemUnfolded); if (!bItemUnfolded) { // Need to supply text. item.pszText = pszText; item.cchTextMax = cchTextMax; ListView_OnGetItem(plv, &item); } } else if (!ListView_IsIconView(plv)) { if (ListView_IsLabelTip(plv) || ListView_IsInfoTip(plv)) { BOOL fSuccess; rcLabel.left = LVIR_LABEL; if (iSubItem) { rcLabel.top = iSubItem; fSuccess = ListView_OnGetSubItemRect(plv, iItem, &rcLabel); } else { fSuccess = ListView_OnGetItemRect(plv, iItem, &rcLabel); } if (fSuccess) { TCHAR szText[INFOTIPSIZE]; item.pszText = szText; item.cchTextMax = min(ARRAYSIZE(szText), cchTextMax); if (ListView_OnGetItem(plv, &item) && item.pszText != LPSTR_TEXTCALLBACK) { SIZE siz; LVFAKEDRAW lvfd; int cx; HRESULT hr = E_FAIL; ListView_BeginFakeCustomDraw(plv, &lvfd, &item); ListView_BeginFakeItemDraw(&lvfd); // ---------Label width----------- ---Client width--- cx = min(rcLabel.right - g_cxLabelMargin, plv->sizeClient.cx); hr = GetTextExtentPoint32(lvfd.nmcd.nmcd.hdc, item.pszText, lstrlen(item.pszText), &siz) ? S_OK : E_FAIL; if (SUCCEEDED(hr) && (rcLabel.left + g_cxLabelMargin + siz.cx) > cx) { lstrcpyn(pszText, item.pszText, item.cchTextMax); } else { // Not truncated after all bItemUnfolded = TRUE; } ListView_EndFakeItemDraw(&lvfd); ListView_EndFakeCustomDraw(&lvfd); } } } } else { // Large icon view is the only one that folds if (ListView_GetUnfoldedRect(plv, iItem, &rcLabel)) { item.pszText = pszText; item.cchTextMax = cchTextMax; ListView_OnGetItem(plv, &item); } else { // Item was never folded bItemUnfolded = TRUE; } } } } return bItemUnfolded; } #ifdef UNICODE // Rather than thunking to ListView_OnGetItemText, we let ListView_GetItemA // do the work. int ListView_OnGetItemTextA(LV* plv, int i, LV_ITEMA *plvi) { if (!plvi) return 0; RIPMSG(plvi->pszText != NULL, "LVM_GETITEMTEXT null string pointer"); plvi->mask = LVIF_TEXT; plvi->iItem = i; if (!ListView_OnGetItemA(plv, plvi)) return 0; return lstrlenA(plvi->pszText); } #endif int ListView_OnGetItemText(LV* plv, int i, LV_ITEM *plvi) { if (!plvi) return 0; RIPMSG(plvi->pszText != NULL, "LVM_GETITEMTEXT null string pointer"); plvi->mask = LVIF_TEXT; plvi->iItem = i; if (!ListView_OnGetItem(plv, plvi)) return 0; return lstrlen(plvi->pszText); } #ifdef UNICODE BOOL WINAPI ListView_OnSetItemTextA(LV* plv, int i, int iSubItem, LPCSTR pszText) { LPWSTR pszW = NULL; BOOL fRet; // Let ListView_OnSetItemText() handle owner-data validation if (pszText != NULL) { pszW = ProduceWFromA(plv->ci.uiCodePage, pszText); if (pszW == NULL) { return FALSE; } } fRet = ListView_OnSetItemText(plv, i, iSubItem, pszW); FreeProducedString(pszW); return fRet; } #endif BOOL WINAPI ListView_OnSetItemText(LV* plv, int i, int iSubItem, LPCTSTR pszText) { LV_ITEM lvi; if (ListView_IsOwnerData(plv)) { RIPMSG(0, "LVM_SETITEMTEXT: Invalid for owner-data listview"); return FALSE; } ListView_InvalidateTTLastHit(plv, i); lvi.mask = LVIF_TEXT; lvi.pszText = (LPTSTR)pszText; lvi.iItem = i; lvi.iSubItem = iSubItem; return ListView_OnSetItem(plv, &lvi); } VOID CALLBACK ImgCtxCallback(void * pvImgCtx, void * pvArg) { LV *plv = (LV *)pvArg; ULONG ulState; SIZE sizeImg; IImgCtx *pImgCtx = plv->pImgCtx; IImgCtx_GetStateInfo(pImgCtx, &ulState, &sizeImg, TRUE); if (ulState & (IMGLOAD_STOPPED | IMGLOAD_ERROR)) { TraceMsg(TF_BKIMAGE, "Listview ImageCallback: Error!"); plv->fImgCtxComplete = FALSE; } else if (ulState & IMGCHG_COMPLETE) { TraceMsg(TF_BKIMAGE, "Listview ImageCallback: Complete!"); plv->fImgCtxComplete = TRUE; InvalidateRect(plv->ci.hwnd, NULL, TRUE); } } void ListView_ReleaseBkImage(LV *plv) { if (plv->pImgCtx) { IImgCtx_Release(plv->pImgCtx); plv->pImgCtx = NULL; if (plv->hpalHalftone) { // No need to delete the half tone palette since we really // share it with the image context and it will clean up. plv->hpalHalftone = NULL; } } if (plv->hbmBkImage) { DeleteObject(plv->hbmBkImage); plv->hbmBkImage = NULL; } if (plv->pszBkImage) { LocalFree(plv->pszBkImage); plv->pszBkImage = NULL; } } BOOL WINAPI ListView_OnSetBkImage(LV* plv, LPLVBKIMAGE pbi) { BOOL fRet = FALSE; if (!pbi) return FALSE; if (pbi->ulFlags & LVBKIF_TYPE_WATERMARK) { BITMAP bm; if (pbi->ulFlags & ~LVBKIF_TYPE_WATERMARK) return FALSE; // We don't support anything else with a watermark if (plv->hbmpWatermark) { DeleteObject(plv->hbmpWatermark); plv->hbmpWatermark = NULL; } if (pbi->hbm && GetObject(pbi->hbm, sizeof(bm), &bm)) { plv->hbmpWatermark = pbi->hbm; plv->szWatermark.cx = bm.bmWidth; plv->szWatermark.cy = bm.bmHeight; fRet = TRUE; } } else { LPCTSTR pszImage = pbi->pszImage; long fl; switch (pbi->ulFlags & LVBKIF_SOURCE_MASK) { case LVBKIF_SOURCE_NONE: TraceMsg(TF_BKIMAGE, "LV SetBkImage to none"); ListView_ReleaseBkImage(plv); break; case LVBKIF_SOURCE_HBITMAP: TraceMsg(TF_BKIMAGE, "LV SetBkImage to hBitmap %08lX", pbi->hbm); ListView_ReleaseBkImage(plv); if (pbi->hbm && (plv->pImgCtx = CBitmapImgCtx_Create(pbi->hbm)) != NULL) { plv->hbmBkImage = pbi->hbm; } else { pbi->ulFlags &= ~LVBKIF_SOURCE_HBITMAP; } break; case LVBKIF_SOURCE_URL: TraceMsg(TF_BKIMAGE, "LV SetBkImage to URL"); ListView_ReleaseBkImage(plv); if (pszImage && pszImage[0]) { HRESULT (*pfnCoCreateInstance)(REFCLSID, IUnknown *, DWORD, REFIID, void * *); HRESULT hr; HMODULE hmodOLE; plv->pszBkImage = LocalAlloc(LPTR, (lstrlen(pszImage) + 1) * sizeof(TCHAR)); if (plv->pszBkImage == NULL) { TraceMsg(TF_BKIMAGE, "Wow, could not allocate memory for string!"); return FALSE; } lstrcpy(plv->pszBkImage, pszImage); if (((hmodOLE = GetModuleHandle(TEXT("OLE32"))) == NULL) || ((pfnCoCreateInstance = (HRESULT (*)(REFCLSID, IUnknown *, DWORD, REFIID, void * *))GetProcAddress(hmodOLE, "CoCreateInstance")) == NULL)) { TraceMsg(TF_BKIMAGE, "Could not find CoCreateInstance!"); TraceMsg(TF_BKIMAGE, "Did the caller remember to call CoInitialize?"); return FALSE; } hr = pfnCoCreateInstance(&CLSID_IImgCtx, NULL, CLSCTX_INPROC_SERVER, &IID_IImgCtx, (void * *)&plv->pImgCtx); if (FAILED(hr)) { TraceMsg(TF_BKIMAGE, "Could not create a pImgCtx!"); TraceMsg(TF_BKIMAGE, "Did you remember to register IEIMAGE.DLL?"); return FALSE; } // // Mirror the downloaded image if the listview window is RTL mirrored, // so that it would be displayed as is. [samera] // fl = ((IS_WINDOW_RTL_MIRRORED(plv->ci.hwnd)) ? DWN_MIRRORIMAGE : 0); hr = IImgCtx_Load(plv->pImgCtx, pszImage, fl); if (FAILED(hr)) { IImgCtx_Release(plv->pImgCtx); plv->pImgCtx = NULL; TraceMsg(TF_BKIMAGE, "Could not init a pImgCtx!"); return FALSE; } } else { pbi->ulFlags &= ~LVBKIF_SOURCE_URL; } break; default: RIPMSG(0, "LVM_SETBKIMAGE: Unsupported image type %d", pbi->ulFlags & LVBKIF_SOURCE_MASK); return FALSE; } plv->ulBkImageFlags = pbi->ulFlags; plv->xOffsetPercent = pbi->xOffsetPercent; plv->yOffsetPercent = pbi->yOffsetPercent; // // If we actually created a pImgCtx, initialize it here. // if (plv->pImgCtx) { if (plv->hpalHalftone == NULL) { IImgCtx_GetPalette(plv->pImgCtx, &plv->hpalHalftone); } plv->fImgCtxComplete = FALSE; IImgCtx_SetCallback(plv->pImgCtx, ImgCtxCallback, plv); IImgCtx_SelectChanges(plv->pImgCtx, IMGCHG_COMPLETE, 0, TRUE); TraceMsg(TF_BKIMAGE, " SUCCESS!"); fRet = TRUE; } } InvalidateRect(plv->ci.hwnd, NULL, TRUE); return fRet; } #ifdef UNICODE BOOL WINAPI ListView_OnSetBkImageA(LV* plv, LPLVBKIMAGEA pbiA) { BOOL fProducedString = FALSE; BOOL fRet; LVBKIMAGEW biW; CopyMemory(&biW, pbiA, sizeof(LVBKIMAGE)); switch (biW.ulFlags & LVBKIF_SOURCE_MASK) { case LVBKIF_SOURCE_NONE: case LVBKIF_SOURCE_HBITMAP: break; case LVBKIF_SOURCE_URL: if (biW.pszImage != NULL) { biW.pszImage = ProduceWFromA(plv->ci.uiCodePage, (LPCSTR)biW.pszImage); if (biW.pszImage == (LPARAM)NULL) { return FALSE; } fProducedString = TRUE; } break; default: // Let ListView_OnSetBkImage() complain about the invalid parameter break; } fRet = ListView_OnSetBkImage(plv, &biW); if (fProducedString) { FreeProducedString((void *)biW.pszImage); } return fRet; } #endif BOOL WINAPI ListView_OnGetBkImage(LV* plv, LPLVBKIMAGE pbi) { BOOL fRet = FALSE; if (!IsBadWritePtr(pbi, sizeof(*pbi))) { if (pbi->ulFlags & LVBKIF_TYPE_WATERMARK) { pbi->hbm = plv->hbmpWatermark; fRet = TRUE; } else { pbi->ulFlags = plv->ulBkImageFlags; switch (plv->ulBkImageFlags & LVBKIF_SOURCE_MASK) { case LVBKIF_SOURCE_NONE: fRet = TRUE; break; case LVBKIF_SOURCE_HBITMAP: pbi->hbm = plv->hbmBkImage; fRet = TRUE; break; case LVBKIF_SOURCE_URL: if (!IsBadWritePtr(pbi->pszImage, pbi->cchImageMax * sizeof(TCHAR))) { lstrcpyn(pbi->pszImage, plv->pszBkImage, pbi->cchImageMax); fRet = TRUE; } break; default: RIPMSG(0, "ListView_OnGetBkImage: Invalid source"); break; } pbi->xOffsetPercent = plv->xOffsetPercent; pbi->yOffsetPercent = plv->yOffsetPercent; } } return fRet; } #ifdef UNICODE BOOL WINAPI ListView_OnGetBkImageA(LV* plv, LPLVBKIMAGEA pbiA) { BOOL fRet = FALSE; if (!IsBadWritePtr(pbiA, sizeof(*pbiA))) { pbiA->ulFlags = plv->ulBkImageFlags; switch (plv->ulBkImageFlags & LVBKIF_SOURCE_MASK) { case LVBKIF_SOURCE_NONE: fRet = TRUE; break; case LVBKIF_SOURCE_HBITMAP: pbiA->hbm = plv->hbmBkImage; fRet = TRUE; break; case LVBKIF_SOURCE_URL: if (!IsBadWritePtr(pbiA->pszImage, pbiA->cchImageMax)) { ConvertWToAN(plv->ci.uiCodePage, pbiA->pszImage, pbiA->cchImageMax, plv->pszBkImage, -1); fRet = TRUE; } break; default: RIPMSG(0, "ListView_OnGetBkImage: Invalid source"); break; } pbiA->xOffsetPercent = plv->xOffsetPercent; pbiA->yOffsetPercent = plv->yOffsetPercent; } return fRet; } #endif void ListView_FreeSubItem(PLISTSUBITEM plsi) { if (plsi) { Str_Set(&plsi->pszText, NULL); LocalFree(plsi); } } int ListView_GetCxScrollbar(LV* plv) { int cx; if (((plv->exStyle & LVS_EX_FLATSB) == 0) || !FlatSB_GetScrollProp(plv->ci.hwnd, WSB_PROP_CXVSCROLL, &cx)) { cx = g_cxScrollbar; } return cx; } int ListView_GetCyScrollbar(LV* plv) { int cy; if (((plv->exStyle & LVS_EX_FLATSB) == 0) || !FlatSB_GetScrollProp(plv->ci.hwnd, WSB_PROP_CYHSCROLL, &cy)) { cy = g_cyScrollbar; } return cy; } DWORD ListView_GetWindowStyle(LV* plv) { DWORD dwStyle; if (((plv->exStyle & LVS_EX_FLATSB) == 0) || !FlatSB_GetScrollProp(plv->ci.hwnd, WSB_PROP_WINSTYLE, (LPINT)&dwStyle)) { dwStyle = GetWindowStyle(plv->ci.hwnd); } return dwStyle; } int ListView_SetScrollInfo(LV *plv, int fnBar, LPSCROLLINFO lpsi, BOOL fRedraw) { int iRc; if (plv->exStyle & LVS_EX_FLATSB) { iRc = FlatSB_SetScrollInfo(plv->ci.hwnd, fnBar, lpsi, fRedraw); } else { iRc = SetScrollInfo(plv->ci.hwnd, fnBar, lpsi, fRedraw); } // // You'd think we were finished, but in fact the game is only half over. // // Some apps (e.g., Font Folder) will do // // SetWindowLong(hwnd, GWL_STYLE, newStyle); // // where newStyle toggles the WS_HSCROLL and/or WS_VSCROLL bits. // This causes USER's internal bookkeeping to go completely out // of whack: The ScrollInfo says that there is a scrollbar, but // the window style says there isn't, or vice versa. The result // is that we get a scrollbar when we shouldn't or vice versa. // // So each time we tweak the scroll info in a manner that changes // the range and page, we kick USER in the head to make sure USER's // view of the world (via style bits) is the same as the scroll // bar's view of the world (via SCROLLINFO). // // // We should always change SIF_PAGE and SIF_RANGE at the same time. // ASSERT((lpsi->fMask & (SIF_PAGE | SIF_RANGE)) == 0 || (lpsi->fMask & (SIF_PAGE | SIF_RANGE)) == (SIF_PAGE | SIF_RANGE)); if ((lpsi->fMask & (SIF_PAGE | SIF_RANGE)) == (SIF_PAGE | SIF_RANGE)) { BOOL fShow; fShow = lpsi->nMax && (int)lpsi->nPage <= lpsi->nMax; #ifdef DEBUG { DWORD dwStyle, dwScroll, dwWant; dwScroll = (fnBar == SB_VERT) ? WS_VSCROLL : WS_HSCROLL; // // We can short-circuit some logic with secret knowledge about how // ListView uses SetScrollInfo. // ASSERT(lpsi->nMin == 0); dwWant = fShow ? dwScroll : 0; dwStyle = ListView_GetWindowStyle(plv); if ((dwStyle & dwScroll) != dwWant) { TraceMsg(TF_LISTVIEW, "ListView_SetScrollInfo: App twiddled WS_[VH]SCROLL"); } } #endif if (plv->exStyle & LVS_EX_FLATSB) FlatSB_ShowScrollBar(plv->ci.hwnd, fnBar, fShow); else ShowScrollBar(plv->ci.hwnd, fnBar, fShow); } return iRc; } // Add/remove/replace item BOOL ListView_FreeItem(LV* plv, LISTITEM* pitem) { ASSERT(!ListView_IsOwnerData(plv)); if (pitem) { if ((pitem->puColumns) && (pitem->cColumns != I_COLUMNSCALLBACK)) LocalFree(pitem->puColumns); Str_Set(&pitem->pszText, NULL); if (pitem->hrgnIcon && pitem->hrgnIcon!=(HANDLE)-1) DeleteObject(pitem->hrgnIcon); // NOTE: We never remove items from the image list; that's // the app's responsibility. // REVIEW: Should we do this? Or should we just provide // a message that will adjust image indices for the guy // when one is removed? // ControlFree(plv->hheap, pitem); } return FALSE; } LISTITEM* ListView_CreateItem(LV* plv, const LV_ITEM* plvi) { LISTITEM* pitem = ControlAlloc(plv->hheap, sizeof(LISTITEM)); ASSERT(!ListView_IsOwnerData(plv)); if (pitem) { if (plvi->mask & LVIF_STATE) { if (plvi->state & ~LVIS_ALL) { DebugMsg(DM_ERROR, TEXT("ListView: Invalid state: %04x"), plvi->state); return NULL; } // If adding a selected item to a single-select listview, deselect // any other items. if ((plv->ci.style & LVS_SINGLESEL) && (plvi->state & LVIS_SELECTED)) ListView_DeselectAll(plv, -1); pitem->state = (plvi->state & ~(LVIS_FOCUSED | LVIS_SELECTED)); } if (plvi->mask & LVIF_PARAM) pitem->lParam = plvi->lParam; if (plvi->mask & LVIF_IMAGE) pitem->iImage = (short) plvi->iImage; if (plvi->mask & LVIF_INDENT) pitem->iIndent = (short) plvi->iIndent; pitem->pt.x = pitem->pt.y = RECOMPUTE; ListView_SetSRecompute(pitem); pitem->pszText = NULL; if (plvi->mask & LVIF_TEXT) { if (!Str_Set(&pitem->pszText, plvi->pszText)) { ListView_FreeItem(plv, pitem); return NULL; } } if ((plvi->mask & LVIF_COLUMNS) && plvi->cColumns) { pitem->cColumns = plvi->cColumns; if (plvi->cColumns != I_COLUMNSCALLBACK) { // Too many columns, or no column array? Then fail. if ((plvi->cColumns > CCMAX_TILE_COLUMNS) || (plvi->puColumns == NULL)) { ListView_FreeItem(plv, pitem); return NULL; } pitem->puColumns = LocalAlloc(LPTR, sizeof(UINT) * pitem->cColumns); if (pitem->puColumns == NULL) { ListView_FreeItem(plv, pitem); return NULL; } CopyMemory(pitem->puColumns, plvi->puColumns, sizeof(UINT) * pitem->cColumns); } } else { pitem->cColumns = 0; pitem->puColumns = NULL; } pitem->dwId = plv->idNext++; // This may overflow. How to deal? } return pitem; } // HACK ALERT!! -- fSmoothScroll is an added parameter! It allows for smooth // scrolling when deleting items. ListView_LRInvalidateBelow is only currently // called from ListView_OnUpdate and ListView_OnDeleteItem. Both these calls // have been modified to work correctly and be backwards compatible. // void ListView_LRInvalidateBelow(LV* plv, int i, int fSmoothScroll) { if (ListView_IsListView(plv) || ListView_IsReportView(plv)) { RECT rcItem; if (!ListView_RedrawEnabled(plv) || (ListView_IsReportView(plv) && (plv->pImgCtx != NULL))) fSmoothScroll = FALSE; if (i >= 0)// && i < ListView_Count(plv)) { ListView_GetRects(plv, i, QUERY_DEFAULT, NULL, NULL, &rcItem, NULL); } else { rcItem.left = rcItem.top = 0; rcItem.right = plv->sizeClient.cx; rcItem.bottom = plv->sizeClient.cy; } // Don't try to scroll over the header part if (ListView_IsReportView(plv) && rcItem.top < plv->yTop) rcItem.top = plv->yTop; // For both List and report view need to erase the item and // below. Note: do simple test to see if there is anything // to redraw // we can't check for bottom/right > 0 because if we nuked something // above or to the left of the view, it may affect us all if ((rcItem.top <= plv->sizeClient.cy) && (rcItem.left <= plv->sizeClient.cx)) { rcItem.bottom = plv->sizeClient.cy; if (ListView_RedrawEnabled(plv)) { if ((plv->clrBk == CLR_NONE) && (plv->pImgCtx == NULL)) { LVSeeThruScroll(plv, &rcItem); } else if (ListView_IsReportView(plv) && fSmoothScroll) { #ifndef UNIX SMOOTHSCROLLINFO si = { sizeof(si), SSIF_MINSCROLL, plv->ci.hwnd, 0, -(plv->cyItem), &rcItem, &rcItem, NULL, NULL, SW_INVALIDATE|SW_ERASE, SSI_DEFAULT, 1, 1, }; #else SMOOTHSCROLLINFO si; si.cbSize = sizeof(si); si.fMask = SSIF_MINSCROLL; si.hwnd = plv->ci.hwnd; si.dx = 0; si.dy = -(plv->cyItem); si.lprcSrc = &rcItem; si.lprcClip = &rcItem; si.hrgnUpdate = NULL; si.lprcUpdate = NULL; si.fuScroll = SW_INVALIDATE|SW_ERASE; si.uMaxScrollTime = SSI_DEFAULT; si.cxMinScroll = 1; si.cyMinScroll = 1; si.pfnScrollProc = NULL; #endif SmoothScrollWindow(&si); } else { RedrawWindow(plv->ci.hwnd, &rcItem, NULL, RDW_INVALIDATE | RDW_ERASE); } } else { RedrawWindow(plv->ci.hwnd, &rcItem, NULL, RDW_INVALIDATE | RDW_ERASE); } if (ListView_IsListView(plv)) { RECT rcClient; // For Listview we need to erase the other columns... rcClient.left = rcItem.right; rcClient.top = 0; rcClient.bottom = plv->sizeClient.cy; rcClient.right = plv->sizeClient.cx; RedrawWindow(plv->ci.hwnd, &rcClient, NULL, RDW_INVALIDATE | RDW_ERASE); } } } } // Used in Ownerdata Icon views to try to not invalidate the whole world... void ListView_IInvalidateBelow(LV* plv, int i) { RECT rcItem; if (i >= 0) { ListView_GetRects(plv, i, QUERY_DEFAULT, NULL, NULL, &rcItem, NULL); } else { rcItem.left = rcItem.top = 0; rcItem.right = plv->sizeClient.cx; rcItem.bottom = plv->sizeClient.cy; } // For Iconviews we need to invalidate everything to the right of us in this // row and everything below the row... // below. Note: do simple test to see if there is anything // to redraw if ((rcItem.top <= plv->sizeClient.cy) && (rcItem.left <= plv->sizeClient.cx)) { rcItem.right = plv->sizeClient.cx; RedrawWindow(plv->ci.hwnd, &rcItem, NULL, RDW_INVALIDATE | RDW_ERASE); // Now erase everything below... rcItem.top = rcItem.bottom; rcItem.bottom = plv->sizeClient.cy; rcItem.left = 0; RedrawWindow(plv->ci.hwnd, &rcItem, NULL, RDW_INVALIDATE | RDW_ERASE); } } void ListView_OnUpdate(LV* plv, int i) { // If in icon/small view, don't call InvalidateItem, since that'll force // FindFreeSlot to get called, which is pig-like. Instead, just // force a WM_PAINT message, which we'll catch and call Recompute with. // if (ListView_IsAutoArrangeView(plv)) { ListView_ArrangeOrSnapToGrid(plv); if (!(plv->ci.style & LVS_AUTOARRANGE)) RedrawWindow(plv->ci.hwnd, NULL, NULL, RDW_INTERNALPAINT | RDW_NOCHILDREN); } else { // HACK ALERT!! -- The third parameter is new. It allows for // smooth scrolling when items are deleted in reportview. // Passing 0, tells it NOT to scroll. // ListView_LRInvalidateBelow(plv, i, 0); } ListView_UpdateScrollBars(plv); } #ifdef UNICODE int ListView_OnInsertItemA(LV* plv, LV_ITEMA* plvi) { LPWSTR pszW = NULL; LPSTR pszC = NULL; int iRet; //HACK ALERT -- this code assumes that LV_ITEMA is exactly the same // as LV_ITEMW except for the pointer to the string. COMPILETIME_ASSERT(sizeof(LV_ITEMA) == sizeof(LV_ITEMW)); if (!plvi) { return -1; } if ((plvi->mask & LVIF_TEXT) && (plvi->pszText != NULL)) { pszC = plvi->pszText; pszW = ProduceWFromA(plv->ci.uiCodePage, pszC); if (pszW == NULL) return -1; plvi->pszText = (LPSTR)pszW; } iRet = ListView_OnInsertItem(plv, (const LV_ITEM*) plvi); if (pszW != NULL) { plvi->pszText = pszC; FreeProducedString(pszW); } return iRet; } #endif int ListView_OnInsertItem(LV* plv, const LV_ITEM* plvi) { int i; ListView_InsertItemInternal(plv, plvi, &i); return i; } LISTITEM* ListView_InsertItemInternal(LV* plv, const LV_ITEM* plvi, int* pi) { int iItem; LISTITEM *pitem = NULL; *pi = -1; if (plvi == NULL) { RIPMSG(0, "ListView_InsertItem: Do not pass a NULL LV_ITEM."); return NULL; } if (plvi->iSubItem != 0) // can only insert the 0th item { RIPMSG(0, "ListView_InsertItem: iSubItem must be 0 (app passed %d)", plvi->iSubItem); return NULL; } // If sorted, then insert sorted. // if (plv->ci.style & (LVS_SORTASCENDING | LVS_SORTDESCENDING) && !ListView_IsOwnerData(plv)) { if (plvi->pszText == LPSTR_TEXTCALLBACK) { DebugMsg(DM_ERROR, TEXT("Don't use LPSTR_TEXTCALLBACK with LVS_SORTASCENDING or LVS_SORTDESCENDING")); return NULL; } iItem = ListView_LookupString(plv, plvi->pszText, LVFI_SUBSTRING | LVFI_NEARESTXY, 0); } else iItem = plvi->iItem; ListView_InvalidateTTLastHit(plv, plv->iTTLastHit); if (!ListView_IsOwnerData(plv)) { int iZ; static s_blah = 0; UINT uSelMask = plvi->mask & LVIF_STATE ? (plvi->state & (LVIS_FOCUSED | LVIS_SELECTED)) : 0; UINT uSel = uSelMask; pitem = ListView_CreateItem(plv, plvi); if (!pitem) return NULL; iItem = DPA_InsertPtr(plv->hdpa, iItem, pitem); if (iItem == -1) { ListView_FreeItem(plv, pitem); return NULL; } plv->cTotalItems++; if (plv->hdpaSubItems) { int iCol; // slide all the colum DPAs down to match the location of the // inserted item // for (iCol = plv->cCol - 1; iCol >= 0; iCol--) { HDPA hdpa = ListView_GetSubItemDPA(plv, iCol); if (hdpa) // this is optional, call backs don't have them { // insert a blank item (REVIEW: should this be callback?) // since this can be a tail sparce array, // we need to make sure enough items are there. if (iItem >= DPA_GetPtrCount(hdpa)) DPA_SetPtr(hdpa, iItem, NULL); else if (DPA_InsertPtr(hdpa, iItem, NULL) != iItem) goto Failure; // Bad assert since hdpa can be tail sparse // ASSERT(ListView_Count(plv) == DPA_GetPtrCount(hdpa)); ASSERT(ListView_Count(plv) >= DPA_GetPtrCount(hdpa)); } } } // Add item to end of z order // iZ = DPA_InsertPtr(plv->hdpaZOrder, ListView_Count(plv), IntToPtr(iItem)); if (iZ == -1) { Failure: DebugMsg(TF_LISTVIEW, TEXT("ListView_OnInsertItem() failed")); if (DPA_DeletePtr(plv->hdpa, iItem)) plv->cTotalItems--; ListView_FreeItem(plv, pitem); return NULL; } // if we inserted before the focus point, move the focus point up one if (iItem <= plv->iFocus) plv->iFocus++; // do the same thing for the mark if (iItem <= plv->iMark) plv->iMark++; // If the item was not added at the end of the list we need // to update the other indexes in the list if (iItem != ListView_Count(plv) - 1) { int i2; for (i2 = iZ - 1; i2 >= 0; i2--) { int iItemZ = (int)(UINT_PTR)DPA_FastGetPtr(plv->hdpaZOrder, i2); if (iItemZ >= iItem) DPA_SetPtr(plv->hdpaZOrder, i2, (void *)(UINT_PTR)(iItemZ + 1)); } } if (ListView_CheckBoxes(plv)) { uSelMask |= LVIS_STATEIMAGEMASK; uSel |= INDEXTOSTATEIMAGEMASK(1); } if (uSelMask) { // we masked off these in the createitem above. // because turning these on means more than setting the bits. ListView_OnSetItemState(plv, iItem, uSel, uSelMask); } if (plvi->mask & LVIF_GROUPID) { int iGroupId = plvi->iGroupId; if (iGroupId == I_GROUPIDNONE) { LISTITEM_SETASKEDFORGROUP(pitem); } else if (iGroupId != I_GROUPIDCALLBACK) { LISTGROUP* pgrp = ListView_FindGroupFromID(plv, iGroupId, NULL); if (!pgrp) { ListView_FreeItem(plv, pitem); return NULL; } pitem->pGroup = pgrp; DPA_AppendPtr(pgrp->hdpa, pitem); } } else { LISTITEM_SETHASNOTASKEDFORGROUP(pitem); } if (plv->fGroupView && (plv->flags & LVF_REDRAW)) { _ListView_RecomputeEx(plv, NULL, 0, FALSE); InvalidateRect(plv->ci.hwnd, NULL, TRUE); } } else { // // simply adjust selection and count // if ((iItem >= 0) && (iItem <= MAX_LISTVIEWITEMS)) { if (FAILED(plv->plvrangeSel->lpVtbl->InsertItem(plv->plvrangeSel, iItem))) { return NULL; } plv->cTotalItems++; plv->rcView.left = RECOMPUTE; ListView_Recompute(plv); if (!ListView_IsReportView(plv) && !ListView_IsListView(plv)) { // We need to erase the background so that we don't leave // turds from wrapped labels in large icon mode. This could // be optimized by only invalidating to the right of and // below the inserted item. InvalidateRect(plv->ci.hwnd, NULL, TRUE); } // if we inserted before the focus point, move the focus point up if (iItem <= plv->iFocus) plv->iFocus++; // do the same thing for the mark if (iItem <= plv->iMark) plv->iMark++; } } if (!ListView_IsOwnerData(plv)) { ASSERT(ListView_Count(plv) == DPA_GetPtrCount(plv->hdpaZOrder)); } if (ListView_RedrawEnabled(plv)) { // Update region ListView_RecalcRegion(plv, TRUE, TRUE); // The Maybe resize colmns may resize things in which case the next call // to Update is not needed. if (!ListView_MaybeResizeListColumns(plv, iItem, iItem)) ListView_OnUpdate(plv, iItem); // this trick makes inserting lots of items cheap // even if redraw is enabled.... don't calc or position items // until this postmessage comes around if (!plv->uUnplaced) { PostMessage(plv->ci.hwnd, LVMI_PLACEITEMS, 0, 0); } plv->uUnplaced++; } else { // // Special case code to make using SetRedraw work reasonably well // for adding items to a listview which is in a non layout mode... // if ((plv->iFirstChangedNoRedraw == -1) || (iItem < plv->iFirstChangedNoRedraw)) plv->iFirstChangedNoRedraw = iItem; } // Nuke insertmark... it may be invalid now that an item has been added. { LVINSERTMARK lvim = {0}; lvim.cbSize = sizeof(LVINSERTMARK); lvim.iItem = -1; ListView_OnSetInsertMark(plv, (LPLVINSERTMARK)&lvim); } ListView_Notify(plv, iItem, 0, LVN_INSERTITEM); NotifyWinEvent(EVENT_OBJECT_CREATE, plv->ci.hwnd, OBJID_CLIENT, iItem+1); *pi = iItem; return pitem; } BOOL ListView_OnDeleteItem(LV* plv, int iItem) { int iCount = ListView_Count(plv); if (!ListView_IsValidItemNumber(plv, iItem)) return FALSE; // out of range NotifyWinEvent(EVENT_OBJECT_DESTROY, plv->ci.hwnd, OBJID_CLIENT, iItem+1); ListView_DismissEdit(plv, TRUE); // cancel edits ListView_OnSetItemState(plv, iItem, 0, LVIS_SELECTED); if (plv->iFocus == iItem) ListView_OnSetItemState(plv, (iItem == iCount - 1 ? iItem - 1 : iItem + 1), LVIS_FOCUSED, LVIS_FOCUSED); ListView_InvalidateTTLastHit(plv, plv->iTTLastHit); if (!ListView_IsOwnerData(plv)) { LISTITEM* pitem = ListView_FastGetItemPtr(plv, iItem); int iZ; if ((plv->rcView.left != RECOMPUTE) && ListView_IsSlotView(plv)) { if (LV_IsItemOnViewEdge(plv, pitem)) { plv->rcView.left = RECOMPUTE; } } ListView_RemoveItemFromItsGroup(plv, pitem); // We don't need to invalidate the item in report view because we // will be scrolling on top of it. // if (!ListView_IsReportView(plv)) ListView_InvalidateItem(plv, iItem, FALSE, RDW_INVALIDATE | RDW_ERASE); // this notify must be done AFTER the Invalidate because some items need callbacks // to calculate the rect, but the notify might free it out ListView_Notify(plv, iItem, 0, LVN_DELETEITEM); // During the notify, the app might've done something to the listview // so revalidate the item number pointer so we don't fault #ifdef DEBUG // Validate internally because DPA_DeletePtr will ASSERT if you ask it // to delete something that doesn't exist. if (!ListView_IsValidItemNumber(plv, iItem)) pitem = NULL; else #endif pitem = DPA_DeletePtr(plv->hdpa, iItem); if (!pitem) { RIPMSG(0, "Something strange happened during LVN_DELETEITEM; abandoning LVM_DELETEITEM"); return FALSE; } plv->cTotalItems = DPA_GetPtrCount(plv->hdpa); // remove from the z-order, this is a lisearch to find this! DPA_DeletePtr(plv->hdpaZOrder, ListView_ZOrderIndex(plv, iItem)); // // As the Z-order hdpa is a set of indexes we also need to decrement // all indexes that exceed the one we are deleting. // for (iZ = ListView_Count(plv) - 1; iZ >= 0; iZ--) { int iItemZ = (int)(UINT_PTR)DPA_FastGetPtr(plv->hdpaZOrder, iZ); if (iItemZ > iItem) DPA_SetPtr(plv->hdpaZOrder, iZ, IntToPtr(iItemZ - 1)); } // remove from sub item DPAs if necessary if (plv->hdpaSubItems) { int iCol; for (iCol = plv->cCol - 1; iCol >= 0; iCol--) { HDPA hdpa = ListView_GetSubItemDPA(plv, iCol); if (hdpa) { // this is optional, call backs don't have them PLISTSUBITEM plsi; // These DPAs are tail sparse, so don't get upset if we // try to delete something that's past the end of the list #ifdef DEBUG plsi = iItem < DPA_GetPtrCount(hdpa) ? DPA_DeletePtr(hdpa, iItem) : NULL; #else plsi = DPA_DeletePtr(hdpa, iItem); #endif ListView_FreeSubItem(plsi); } } } ListView_FreeItem(plv, pitem); // ... finaly the item pointer if (plv->fGroupView && (plv->flags & LVF_REDRAW)) { _ListView_RecomputeEx(plv, NULL, 0, TRUE); InvalidateRect(plv->ci.hwnd, NULL, TRUE); } } else { // // simply notify and then fixup selection state and count // if ((iItem >= 0) && (iItem <= MAX_LISTVIEWITEMS)) { ListView_Notify(plv, iItem, 0, LVN_DELETEITEM); if (FAILED(plv->plvrangeSel->lpVtbl->RemoveItem(plv->plvrangeSel, iItem))) { SetLastError(ERROR_OUTOFMEMORY); return FALSE; } plv->cTotalItems--; plv->rcView.left = RECOMPUTE; ListView_Recompute(plv); if (!ListView_IsReportView(plv) && !ListView_IsListView(plv)) { // We need to erase the background so that the last item gets // erased in both icon modes and so that we don't leave turds // from wrapped labels in large icon mode. This could be // optimized by only invalidating to the right of and below // the deleted item. InvalidateRect(plv->ci.hwnd, NULL, TRUE); } } else { return FALSE; } } iCount = ListView_Count(plv); // regrab count incase someone updated item... if (!ListView_IsOwnerData(plv)) { ASSERT(ListView_Count(plv) == DPA_GetPtrCount(plv->hdpaZOrder)); } if (plv->iFocus == iItem) { if (plv->iFocus >= iCount) { plv->iFocus = iCount - 1; } } if (plv->iFocus > iItem) { plv->iFocus--; // slide the focus index down } // same with the mark if (plv->iMark == iItem) { // deleted the mark item if (plv->iMark >= iCount) // did we nuke the last item? plv->iMark = iCount - 1; } else if (plv->iMark > iItem) plv->iMark--; // slide the mark index down // Free up the hot item if (plv->iHot == iItem) plv->iHot = -1; // Deleting an icon invalidates the icon positioning cache plv->iFreeSlot = -1; // HACK ALERT!! -- This construct with ReportView steals code from // ListView_OnUpdate. Currently, it will work exactly the same as before, // EXCEPT, that it won't call ListView_OnUpdate. This is to allow us to // send a flag to ListView_LRUpdateBelow to tell it we're scrolling up. // if (ListView_IsReportView(plv)) { // if the new count is zero and we will be showing empty text, simply invalidate the // rect and redraw, else go through the invalidate below code... // we don't know if we are going to show empty text if pszEmptyText is NULL, or not // because we may get one through notify, so if iCount is 0 invalidate everything if (iCount == 0) InvalidateRect(plv->ci.hwnd, NULL, TRUE); else ListView_LRInvalidateBelow(plv,iItem,1); if (ListView_RedrawEnabled(plv)) ListView_UpdateScrollBars(plv); else { // // Special case code to make using SetRedraw work reasonably well // for adding items to a listview which is in a non layout mode... // if ((plv->iFirstChangedNoRedraw != -1) && (iItem < plv->iFirstChangedNoRedraw)) plv->iFirstChangedNoRedraw--; } } else { if (ListView_RedrawEnabled(plv)) ListView_OnUpdate(plv, iItem); else { ListView_LRInvalidateBelow(plv, iItem, 0); // // Special case code to make using SetRedraw work reasonably well // for adding items to a listview which is in a non layout mode... // if ((plv->iFirstChangedNoRedraw != -1) && (iItem < plv->iFirstChangedNoRedraw)) plv->iFirstChangedNoRedraw--; } } ListView_RecalcRegion(plv, TRUE, TRUE); return TRUE; } void ListView_DeleteAllGroupItems(LV* plv) { if (plv->hdpaGroups) { int iGroup, cGroups = DPA_GetPtrCount(plv->hdpaGroups); for (iGroup = 0; iGroup < cGroups; iGroup++) { LISTGROUP* pgrp = DPA_FastGetPtr(plv->hdpaGroups, iGroup); DPA_Destroy(pgrp->hdpa); pgrp->hdpa = DPA_Create(5); } } } BOOL ListView_OnDeleteAllItems(LV* plv) { int i; BOOL bAlreadyNotified; BOOL fHasItemData = !ListView_IsOwnerData(plv); ListView_DismissEdit(plv, TRUE); // cancel edits ListView_DeleteAllGroupItems(plv); // Must neutralize the focus because some apps will call // ListView_OnGetNextItem(LVNI_FOCUSED) during delete notifications, // so we need to make sure the focus is in a safe place. // May as well neutralize the mark, too. plv->iMark = plv->iFocus = -1; // Also nuke the icon positioning cache plv->iFreeSlot = -1; // Since we delete all items, There is no insertion slot! plv->iInsertItem = -1; bAlreadyNotified = (BOOL)ListView_Notify(plv, -1, 0, LVN_DELETEALLITEMS); ListView_InvalidateTTLastHit(plv, plv->iTTLastHit); if (fHasItemData || !bAlreadyNotified) { for (i = ListView_Count(plv) - 1; i >= 0; i--) { if (!bAlreadyNotified) ListView_Notify(plv, i, 0, LVN_DELETEITEM); if (fHasItemData) { ListView_FreeItem(plv, ListView_FastGetItemPtr(plv, i)); // // CAREFUL! Applications such as NT Backup call back // into ListView during the LVN_DELETEITEM notification, // so we need to kill this item or we will fault at the // next iteration because everybody relies on // ListView_Count for validation. // DPA_FastDeleteLastPtr(plv->hdpa); plv->cTotalItems--; } } } if (ListView_IsOwnerData(plv)) { if (FAILED(plv->plvrangeSel->lpVtbl->Clear(plv->plvrangeSel))) { SetLastError(ERROR_OUTOFMEMORY); } plv->cTotalItems = 0; } else { DPA_DeleteAllPtrs(plv->hdpa); DPA_DeleteAllPtrs(plv->hdpaZOrder); plv->cTotalItems = 0; if (plv->hdpaSubItems) { int iCol; for (iCol = plv->cCol - 1; iCol >= 0; iCol--) { HDPA hdpa = ListView_GetSubItemDPA(plv, iCol); if (hdpa) { DPA_EnumCallback(hdpa, ListView_FreeColumnData, 0); DPA_DeleteAllPtrs(hdpa); } } } } plv->rcView.left = RECOMPUTE; plv->xOrigin = 0; plv->nSelected = 0; plv->ptlRptOrigin.x = 0; plv->ptlRptOrigin.y = 0; // reset the cxItem width if (!(plv->flags & LVF_COLSIZESET)) { plv->cxItem = ListView_ComputeCXItemSize(plv); } RedrawWindow(plv->ci.hwnd, NULL, NULL, RDW_INVALIDATE | RDW_ERASE); ListView_UpdateScrollBars(plv); return TRUE; } int ListView_IFindNearestItem(LV* plv, int left, int top, UINT vk) { int iMin = -1; if (ListView_IsOwnerData(plv)) { POINT pt; int cSlots; int iWidth = 0, iHeight = 0; ASSERT(!ListView_IsReportView(plv) && !ListView_IsListView(plv)); pt.x = left + plv->ptOrigin.x; pt.y = top + plv->ptOrigin.y; cSlots = ListView_GetSlotCount(plv, TRUE, &iWidth, &iHeight); iMin = ListView_CalcHitSlot(plv, pt, cSlots, iWidth, iHeight); switch(vk) { case VK_HOME: iMin = 0; break; case VK_END: iMin = ListView_Count(plv) - 1; break; case VK_LEFT: if (iMin % cSlots) iMin -= 1; break; case VK_RIGHT: if ((iMin + 1) % cSlots) iMin += 1; break; case VK_UP: if (iMin >= cSlots) iMin -= cSlots; break; case VK_DOWN: if (iMin + cSlots < ListView_Count(plv)) iMin += cSlots; break; default: ; } iMin = max(0, iMin); iMin = min(ListView_Count(plv) - 1, iMin); } else { ULONGLONG dMin = 0; int cyItem; int yEnd = 0, yLimit = 0, xEnd = 0; int iCount; int i; if (ListView_UseLargeIcons(plv)) { cyItem = plv->cyIcon; } else { cyItem = plv->cyItem; } iCount = ListView_Count(plv); if (iCount == 1) return 0; if (vk == VK_HOME) { yEnd = yLimit = plv->rcView.bottom; xEnd = plv->rcView.right; } else if (vk == VK_END) { yEnd = yLimit = plv->rcView.top; xEnd = plv->rcView.left; } for (i = 0; i < iCount; i++) { RECT rc; int dx; ULONGLONG dxAbs, dyAbs, dOffset; int dy; ListView_GetRects(plv, i, QUERY_DEFAULT, &rc, NULL, NULL, NULL); dx = rc.left - left; dxAbs = (ULONGLONG)(dx < 0 ? -dx : dx); dy = rc.top - top; dyAbs = (ULONGLONG)(dy < 0 ? -dy : dy); if ((vk == VK_LEFT) && (dxAbs < dyAbs || dx >= 0)) continue; else if ((vk == VK_RIGHT) && (dxAbs < dyAbs || dx <= 0)) continue; else if ((vk == VK_UP) && (dxAbs > dyAbs || dy >= 0)) continue; else if ((vk == VK_DOWN) && (dxAbs > dyAbs || dy <= 0)) continue; if (vk == VK_HOME || vk == VK_END) { // home is not the nearest to the top corner, it's the leftmost of the top row. // ditto (reversed) for end. thus we can't use the stuff below. bummer if (vk == VK_HOME) { if ((rc.top + cyItem < yEnd) || // if it's fully above the highest line so, take it! ((rc.top < yLimit) && // if it's on the same row as the top item to date (rc.left < xEnd))) { iMin = i; xEnd = rc.left; yEnd = rc.top; if (rc.top + cyItem < yLimit) yLimit = rc.top + cyItem; } } else { if ((rc.top > yEnd) || //if it's full below the lowest row ((rc.top + cyItem > yLimit) && // if it's on the same row (rc.right > xEnd))) { iMin = i; xEnd = rc.right; yEnd = rc.top; if (rc.top > yLimit) yLimit = rc.top; } } } else { dOffset = ((dxAbs * dxAbs) + (dyAbs * dyAbs)); if (iMin == -1 || (dMin > dOffset)) { dMin = dOffset; iMin = i; } } } } return iMin; } int ListView_Arrow(LV* plv, int iStart, UINT vk) { RECT rcFocus; int i; int dx; int iCount; // // The algorithm to find which item depends if we are in a view // that is arrange(layout) oriented or a sorted (list) view. // For the sorted views we will use some optimizations to make // it faster // iCount = ListView_Count(plv); if ((ListView_IsReportView(plv) || ListView_IsListView(plv)) && !plv->fGroupView) { // // For up and down arrows, simply increment or decrement the // index. Note: in listview this will cause it to wrap columns // which is fine as it is compatible with the file manager // // Assumes only one of these flags is set... switch (vk) { case VK_LEFT: if (ListView_IsReportView(plv)) { ListView_ROnScroll(plv, (GetAsyncKeyState(VK_CONTROL) < 0) ? SB_PAGELEFT : SB_LINELEFT, 0, SB_HORZ); } else iStart -= plv->cItemCol; break; case VK_RIGHT: if (ListView_IsReportView(plv)) { // Make this horizontally scroll the report view ListView_ROnScroll(plv, (GetAsyncKeyState(VK_CONTROL) < 0) ? SB_PAGERIGHT : SB_LINERIGHT, 0, SB_HORZ); } else iStart += plv->cItemCol; break; case VK_UP: iStart--; break; case VK_DOWN: iStart++; break; case VK_HOME: iStart = 0; break; case VK_END: iStart = iCount -1; break; case VK_NEXT: if (ListView_IsReportView(plv)) { i = iStart; // save away to make sure we dont go wrong way! // First go to end of page... iStart = (int)(((LONG)(plv->sizeClient.cy - (plv->cyItem) - plv->yTop) + plv->ptlRptOrigin.y) / plv->cyItem); // If Same item, increment by page size. if (iStart <= i) iStart = i + max( (plv->sizeClient.cy - plv->yTop)/ plv->cyItem - 1, 1); if (iStart >= iCount) iStart = iCount - 1; } else { // multiply by 2/3 to give a good feel.. when the item is mostly shown // you want to go to the next column dx = (plv->sizeClient.cx + (plv->cxItem*2)/3) / plv->cxItem; if (!dx) dx = 1; iStart += plv->cItemCol * dx; if (plv->cItemCol) { while (iStart >= iCount) iStart -= plv->cItemCol; } } break; case VK_PRIOR: if (ListView_IsReportView(plv)) { i = iStart; // save away to make sure we dont go wrong way! // First go to end of page... iStart = (int)(plv->ptlRptOrigin.y / plv->cyItem); // If Same item, increment by page size. if (iStart >= i) iStart = i - max( (plv->sizeClient.cy - plv->yTop)/ plv->cyItem - 1, 1); if (iStart < 0) iStart = 0; } else { dx = (plv->sizeClient.cx + (plv->cxItem*2)/3) / plv->cxItem; if (!dx) dx = 1; iStart -= plv->cItemCol * dx; if (plv->cItemCol) { while (iStart < 0) iStart += plv->cItemCol; } } break; default: return -1; // Out of range } // Make sure it is in range!. if ((iStart >= 0) && (iStart < iCount)) return iStart; else if (iCount == 1) return 0; else return -1; } else { // // Layout type view. we need to use the position of the items // to figure out the next item // if (ListView_IsOwnerData(plv)) { iStart = max(0, iStart); // if it does not matches any of the entries in the case statement below // this is done to skip the call back by the GetRects // if (vk != VK_LEFT && vk != VK_RIGHT && vk != VK_UP && vk != VK_DOWN && vk != VK_HOME && vk != VK_END && vk != VK_NEXT && vk != VK_PRIOR) { return -1; } ListView_GetRects(plv, iStart, QUERY_DEFAULT, &rcFocus, NULL, NULL, NULL); } else { if (iStart != -1) { ListView_GetRects(plv, iStart, QUERY_DEFAULT, &rcFocus, NULL, NULL, NULL); } } switch (vk) { // For standard arrow keys just fall out of here. case VK_LEFT: case VK_RIGHT: case VK_UP: case VK_DOWN: if (ListView_IsOwnerData(plv)) { break; } else { if (iStart != -1) { // all keys map to VK_HOME except VK_END break; } // Fall through vk = VK_HOME; } case VK_HOME: rcFocus.left = - plv->ptOrigin.x; rcFocus.top = - plv->ptOrigin.y; break; case VK_END: rcFocus.left = plv->rcView.right; rcFocus.top = plv->rcView.bottom; break; case VK_NEXT: rcFocus.top += plv->sizeClient.cy; vk = VK_UP; break; case VK_PRIOR: vk = VK_DOWN; rcFocus.top -= plv->sizeClient.cy; break; default: return -1; // Out of range } return ListView_IFindNearestItem(plv, rcFocus.left, rcFocus.top, vk); } } int ListView_OnGetNextItem(LV* plv, int i, UINT flags) { int iStart = i; int cItemMax = ListView_Count(plv); // Note that -1 is a valid starting point if (i < -1 || i >= cItemMax) return -1; if (ListView_IsOwnerData(plv)) { if (flags & (LVNI_CUT | LVNI_DROPHILITED | LVNI_PREVIOUS)) { return -1; } } if (flags & LVNI_FOCUSED) { // we know which item is focused, jump right to it. // but we have to mimick the code below exactly for compat: // if directional bits are set, they take precedence. if (!(flags & (LVNI_ABOVE | LVNI_BELOW | LVNI_TORIGHT | LVNI_TOLEFT))) { // there are no more focused items after iFocus if (i >= plv->iFocus) return -1; // subtract one here -- we increment it below i = plv->iFocus - 1; } } while (TRUE) { if (flags & (LVNI_ABOVE | LVNI_BELOW | LVNI_TORIGHT | LVNI_TOLEFT)) { UINT vk; if (flags & LVNI_ABOVE) vk = VK_UP; else if (flags & LVNI_BELOW) vk = VK_DOWN; else if (flags & LVNI_TORIGHT) vk = VK_RIGHT; else vk = VK_LEFT; if (i != -1) i = ListView_Arrow(plv, i, vk); if (i == -1) return i; } else { i++; if (i == cItemMax) return -1; } // See if any other restrictions are set if (flags & ~(LVNI_ABOVE | LVNI_BELOW | LVNI_TORIGHT | LVNI_TOLEFT)) { WORD wItemState; if (ListView_IsOwnerData(plv)) { if (flags & LVNI_FOCUSED) { // we check LVNI_FOCUSED before the loop, so i == iFocus ASSERT(i == plv->iFocus && i != -1); if (flags & LVNI_SELECTED) { if (plv->plvrangeSel->lpVtbl->IsSelected(plv->plvrangeSel, i) != S_OK) { i = -1; } } } else if (flags & LVNI_SELECTED) { i = max(i, 0); plv->plvrangeSel->lpVtbl->NextSelected(plv->plvrangeSel, i, &i); } else { i = -1; } } else { { LISTITEM* pitem = ListView_FastGetItemPtr(plv, i); wItemState = pitem->state; } // for LVNI_FOCUSED, we start at the LVIS_FOCUSED element, if we're // not on that element, one of the below continues was hit, so // we'll never find the element. bail out early. if ((flags & LVNI_FOCUSED) && !(wItemState & LVIS_FOCUSED)) { ASSERT(i == plv->iFocus || i == plv->iFocus+1); return -1; } if (((flags & LVNI_SELECTED) && !(wItemState & LVIS_SELECTED)) || ((flags & LVNI_CUT) && !(wItemState & LVIS_CUT)) || ((flags & LVNI_DROPHILITED) && !(wItemState & LVIS_DROPHILITED))) { if (i != iStart) continue; else { // we've looped and we can't find anything to fit this criteria return -1; } } } } return i; } } int ListView_CompareString(LV* plv, int i, LPCTSTR pszFind, UINT flags, int iLen) { // REARCHITECT: non protected globals int cb; TCHAR ach[CCHLABELMAX]; LV_ITEM item; ASSERT(!ListView_IsOwnerData(plv)); ASSERT(pszFind); item.iItem = i; item.iSubItem = 0; item.mask = LVIF_TEXT; item.pszText = ach; item.cchTextMax = ARRAYSIZE(ach); ListView_OnGetItem(plv, &item); if (!(flags & (LVFI_PARTIAL | LVFI_SUBSTRING))) return lstrcmpi(item.pszText, pszFind); // FEATURE: LVFI_SUBSTRING is not really implemented yet. cb = lstrlen(pszFind); if (iLen && (cb > iLen)) { cb = iLen; } // // If the sub strings not equal then return the ordering based // on the entire string. // #ifndef UNIX return IntlStrEqNI(item.pszText, pszFind, cb) ? 0 : lstrcmp(item.pszText, pszFind); #else return IntlStrEqNI(item.pszText, pszFind, cb) ? 0 : lstrcmpi(item.pszText, pszFind); #endif } #ifdef UNICODE int ListView_OnFindItemA(LV* plv, int iStart, LV_FINDINFOA * plvfi) { LPWSTR pszW = NULL; LPCSTR pszC = NULL; int iRet; //HACK ALERT -- this code assumes that LV_FINDINFOA is exactly the same // as LV_FINDINFOW except for the pointer to the string. COMPILETIME_ASSERT(sizeof(LV_FINDINFOA) == sizeof(LV_FINDINFOW)); if (!plvfi) return -1; if (!(plvfi->flags & LVFI_PARAM) && !(plvfi->flags & LVFI_NEARESTXY)) { pszC = plvfi->psz; if ((pszW = ProduceWFromA(plv->ci.uiCodePage, pszC)) == NULL) return -1; plvfi->psz = (LPSTR)pszW; } iRet = ListView_OnFindItem(plv, iStart, (const LV_FINDINFO *)plvfi); if (pszW != NULL) { plvfi->psz = pszC; FreeProducedString(pszW); } return iRet; } #endif int ListView_OnFindItem(LV* plv, int iStart, const LV_FINDINFO* plvfi) { int i; int j; int cItem; UINT flags; if (!plvfi) return -1; if (plvfi->flags & LVFI_NEARESTXY) { if (ListView_IsSlotView(plv)) { return ListView_IFindNearestItem(plv, plvfi->pt.x, plvfi->pt.y, plvfi->vkDirection); } else return -1; } // Note that -1 is a valid starting point if (iStart < -1 || iStart >= ListView_Count(plv)) return -1; if (ListView_IsOwnerData(plv)) { // call back to owner for search return (int) ListView_RequestFindItem(plv, plvfi, iStart + 1); } else { flags = plvfi->flags; i = iStart; cItem = ListView_Count(plv); if (flags & LVFI_PARAM) { LPARAM lParam = plvfi->lParam; // Lisearch with wraparound... // for (j = cItem; j-- != 0;) { ++i; if (i == cItem) { if (flags & LVFI_WRAP) i = 0; else break; } if (ListView_FastGetItemPtr(plv, i)->lParam == lParam) return i; } } else // if (flags & (LVFI_STRING | LVFI_SUBSTRING | LVFI_PARTIAL)) { LPCTSTR pszFind = plvfi->psz; if (!pszFind) return -1; if (plv->ci.style & (LVS_SORTASCENDING | LVS_SORTDESCENDING)) return ListView_LookupString(plv, pszFind, flags, i + 1); for (j = cItem; j-- != 0;) { ++i; if (i == cItem) { if (flags & LVFI_WRAP) i = 0; else break; } if (ListView_CompareString(plv, i, pszFind, (flags & (LVFI_PARTIAL | LVFI_SUBSTRING)), 0) == 0) { return i; } } } } return -1; } BOOL ListView_OnGetItemRect(LV* plv, int i, RECT* prc) { LPRECT pRects[LVIR_MAX]; // validate parameters if (!ListView_IsValidItemNumber(plv, i)) { RIPMSG(0, "LVM_GETITEMRECT: invalid index %d", i); return FALSE; } if (!prc || prc->left >= LVIR_MAX || prc->left < 0) { RIPMSG(0, "LVM_GETITEMRECT: invalid rect pointer"); return FALSE; } pRects[0] = NULL; pRects[1] = NULL; pRects[2] = NULL; pRects[3] = NULL; pRects[prc->left] = prc; ListView_GetRects(plv, i, QUERY_DEFAULT, pRects[LVIR_ICON], pRects[LVIR_LABEL], pRects[LVIR_BOUNDS], pRects[LVIR_SELECTBOUNDS]); return TRUE; } // // in: // plv // iItem MUST be a valid item index (in range) // out: // prcIcon icon bounding rect // prcLabel label text bounding rect, for details this is the first column // prcBounds entire item (all text and icon), including columns in details // prcSelectionBounds union of icon and label rects, does NOT include columns // in details view // REARCHITECT raymondc - Need to pass an HDC parameter for measurement // since sometimes we do this while painting // This returns rects in Window Coordinates void ListView_GetRects(LV* plv, int iItem, UINT fQueryLabelRects, RECT* prcIcon, RECT* prcLabel, RECT* prcBounds, RECT* prcSelectBounds) { ASSERT(plv); if (ListView_IsReportView(plv)) { ListView_RGetRects(plv, iItem, prcIcon, prcLabel, prcBounds, prcSelectBounds); } else if (ListView_IsListView(plv)) { ListView_LGetRects(plv, iItem, prcIcon, prcLabel, prcBounds, prcSelectBounds); } else { if (ListView_IsOwnerData(plv)) { RECT rcIcon; RECT rcTextBounds; LISTITEM item; if (ListView_IsIconView(plv)) ListView_IGetRectsOwnerData(plv, iItem, &rcIcon, &rcTextBounds, &item, FALSE); else if (ListView_IsSmallView(plv)) ListView_SGetRectsOwnerData(plv, iItem, &rcIcon, &rcTextBounds, &item, FALSE); else if (ListView_IsTileView(plv)) ListView_TGetRectsOwnerData(plv, iItem, &rcIcon, &rcTextBounds, &item, FALSE); if (prcIcon) *prcIcon = rcIcon; if (prcLabel) *prcLabel = rcTextBounds; if (prcBounds) UnionRect(prcBounds, &rcIcon, &rcTextBounds); if (prcSelectBounds) UnionRect(prcSelectBounds, &rcIcon, &rcTextBounds); } else { if (iItem >= ListView_Count(plv)) { return; } else { LISTITEM *pitem = ListView_FastGetItemPtr(plv, iItem); if (pitem->cyFoldedLabel == SRECOMPUTE) { _ListView_RecomputeLabelSize(plv, pitem, iItem, NULL, FALSE); } _ListView_GetRectsFromItem(plv, ListView_IsSmallView(plv), pitem, fQueryLabelRects, prcIcon, prcLabel, prcBounds, prcSelectBounds); } } } } void ListView_GetRectsOwnerData(LV* plv, int iItem, RECT* prcIcon, RECT* prcLabel, RECT* prcBounds, RECT* prcSelectBounds, LISTITEM* pitem) { ASSERT(plv); ASSERT(ListView_IsOwnerData(plv)); if (ListView_IsReportView(plv)) { ListView_RGetRects(plv, iItem, prcIcon, prcLabel, prcBounds, prcSelectBounds); } else if (ListView_IsListView(plv)) { ListView_LGetRects(plv, iItem, prcIcon, prcLabel, prcBounds, prcSelectBounds); } else { RECT rcIcon; RECT rcTextBounds; if (ListView_IsIconView(plv)) ListView_IGetRectsOwnerData(plv, iItem, &rcIcon, &rcTextBounds, pitem, TRUE); else if (ListView_IsSmallView(plv)) ListView_SGetRectsOwnerData(plv, iItem, &rcIcon, &rcTextBounds, pitem, TRUE); else if (ListView_IsTileView(plv)) ListView_TGetRectsOwnerData(plv, iItem, &rcIcon, &rcTextBounds, pitem, TRUE); // Don't need to check for folding here, as will have been handled in user data // rectangle fetching functions. if (prcIcon) *prcIcon = rcIcon; if (prcLabel) *prcLabel = rcTextBounds; if (prcBounds) UnionRect(prcBounds, &rcIcon, &rcTextBounds); if (prcSelectBounds) UnionRect(prcSelectBounds, &rcIcon, &rcTextBounds); } } BOOL ListView_OnRedrawItems(LV* plv, int iFirst, int iLast) { int iCount = ListView_Count(plv); if (iFirst < iCount) { if (iLast >= iCount) iLast = iCount - 1; while (iFirst <= iLast) ListView_InvalidateItem(plv, iFirst++, FALSE, RDW_INVALIDATE | RDW_ERASE); } return TRUE; } // fSelectionOnly use the selection bounds only, ie. don't include // columns in invalidation if in details view // void ListView_InvalidateItemEx(LV* plv, int iItem, BOOL fSelectionOnly, UINT fRedraw, UINT maskChanged) { RECT rc; LPRECT prcIcon; LPRECT prcLabel; LPRECT prcBounds; LPRECT prcSelectBounds; LISTITEM* pitem = NULL; if (iItem == -1) return; // Ok if NULL if (plv->hdpa) pitem = ListView_GetItemPtr(plv, iItem); prcIcon = prcLabel = prcBounds = prcSelectBounds = NULL; // if we're in owner draw mode, and there's been a new font, // we don't really know what the selection bounds is, so always use the bounds // in that case... unless we're in fullrowselect mode if (ListView_IsOwnerData(plv) && plv->flags & LVF_CUSTOMFONT && !ListView_FullRowSelect(plv)) { fSelectionOnly = FALSE; } // if we're owner draw, there's no such thing as selection only if (plv->ci.style & LVS_OWNERDRAWFIXED) fSelectionOnly = FALSE; if (fSelectionOnly) { // In report mode non-fullrowselect, // we have to use the full label rectangle rather // than just the selection bounds, since the stuff outside the // rectangle might need redrawing, too. if (ListView_IsReportView(plv) && !ListView_FullRowSelect(plv)) prcLabel = &rc; else prcSelectBounds = &rc; } else { // if _only_the_text_ or _only_the_image_ changed then limit the redraw switch (maskChanged) { case LVIF_IMAGE: prcIcon = &rc; break; case LVIF_TEXT: prcLabel = &rc; break; default: prcBounds = &rc; break; } } if (ListView_RedrawEnabled(plv)) { ListView_GetRects(plv, iItem, QUERY_DEFAULT, prcIcon, prcLabel, prcBounds, prcSelectBounds); if (RECTS_IN_SIZE(plv->sizeClient, rc)) { if (ListView_IsBorderSelect(plv)) { InflateRect(&rc, 4 + g_cxIconMargin, 4 + g_cyIconMargin); // account for selection border and seperation since drawing otside of icon fRedraw |= RDW_ERASE; } // Affects only allowed if dubble buffering if (ListView_IsDoubleBuffer(plv)) { if ((pitem && (pitem->state & LVIS_GLOW))) { InflateRect(&rc, GLOW_EXPAND, GLOW_EXPAND); fRedraw |= RDW_ERASE; } } ListView_DebugDrawInvalidRegion(plv, &rc, NULL); RedrawWindow(plv->ci.hwnd, &rc, NULL, fRedraw); } } else { // if we're not visible, we'll get a full // erase bk when we do become visible, so only do this stuff when // we're on setredraw false if (!(plv->flags & LVF_REDRAW)) { // if we're invalidating that's new (thus hasn't been painted yet) // blow it off if ((plv->iFirstChangedNoRedraw != -1) && (iItem >= plv->iFirstChangedNoRedraw)) { return; } ListView_GetRects(plv, iItem, QUERY_DEFAULT, prcIcon, prcLabel, prcBounds, prcSelectBounds); // Affects only allowed if dubble buffering if (ListView_IsDoubleBuffer(plv)) { if (pitem && (pitem->state & LVIS_GLOW)) { InflateRect(&rc, GLOW_EXPAND, GLOW_EXPAND); fRedraw |= RDW_ERASE; } } if (ListView_IsBorderSelect(plv)) { InflateRect(&rc, 4 + g_cxIconMargin, 4 + g_cyIconMargin); // account for selection border and seperation since drawing otside of icon fRedraw |= RDW_ERASE; } // if it had the erase bit, add it to our region if (RECTS_IN_SIZE(plv->sizeClient, rc)) { HRGN hrgn = CreateRectRgnIndirect(&rc); ListView_InvalidateRegion(plv, hrgn); if (fRedraw & RDW_ERASE) plv->flags |= LVF_ERASE; } } } } // this returns BF_* flags to indicate which if any edge the item I is touching // or crossing... UINT LV_IsItemOnViewEdge(LV* plv, LISTITEM* pitem) { RECT rcItem; UINT uRet = 0; // as far as rcView goes, unfolded label rects determine edge-ness _ListView_GetRectsFromItem(plv, ListView_IsSmallView(plv), pitem, QUERY_RCVIEW|QUERY_UNFOLDED, NULL, NULL, &rcItem, NULL); // translate from window coordinates to listview coordinate OffsetRect(&rcItem, plv->ptOrigin.x, plv->ptOrigin.y); // include the rcView buffer ListView_AddViewRectBuffer(plv, &rcItem); if (rcItem.right >= plv->rcView.right) uRet |= BF_RIGHT; if (rcItem.left <= plv->rcView.left) uRet |= BF_LEFT; if (rcItem.top <= plv->rcView.top) uRet |= BF_TOP; if (rcItem.bottom >= plv->rcView.bottom) uRet |= BF_BOTTOM; return uRet; } // Move pitem to x,y // Update rcView to accomodate this if we can, or mark rcView for recomputation void LV_AdjustViewRectOnMove(LV* plv, LISTITEM *pitem, int x, int y) { plv->iFreeSlot = -1; // The "free slot" cache is no good once an item moves // if we have to recompute anyways, don't bother if (!ListView_IsOwnerData(plv)) { if ((plv->rcView.left != RECOMPUTE) && x != RECOMPUTE && y != RECOMPUTE && pitem->cyFoldedLabel != SRECOMPUTE) { RECT rcClient, rcAfter; RECT rcView = plv->rcView; // Our optimized move-adjust-rcView must maintain this, make sure it's true before we even start: ASSERT(ListView_ValidatercView(plv, &plv->rcView, FALSE)); ListView_GetClientRect(plv, &rcClient, TRUE, NULL); ASSERT(ListView_ValidateScrollPositions(plv, &rcClient)); if (pitem->pt.x != RECOMPUTE) { UINT uEdges; uEdges = LV_IsItemOnViewEdge(plv, pitem); pitem->pt.x = x; pitem->pt.y = y; // before and after the move, they need to be touching the // same edges or not at all if (uEdges != LV_IsItemOnViewEdge(plv, pitem)) { goto FullRecompute; } } else { // if the position wasn't set before // we just need to find out what it is afterwards and // enlarge the view... we don't need to shrink it pitem->pt.x = x; pitem->pt.y = y; } _ListView_GetRectsFromItem(plv, ListView_IsSmallView(plv), pitem, QUERY_RCVIEW|QUERY_UNFOLDED, NULL, NULL, &rcAfter, NULL); // translate from window coordinates to listview coordinates OffsetRect(&rcAfter, plv->ptOrigin.x, plv->ptOrigin.y); // include the rcView buffer ListView_AddViewRectBuffer(plv, &rcAfter); // if we make it here, we just have to make sure the new view rect // encompases this new item UnionRect(&rcView, &rcView, &rcAfter); DebugMsg(TF_LISTVIEW, TEXT("Score! (%d %d %d %d) was (%d %d %d %d)"), rcView.left, rcView.top, rcView.right, rcView.bottom, plv->rcView.left, plv->rcView.top, plv->rcView.right, plv->rcView.bottom); // Our optimized move-adjust-rcView must maintain this: ASSERT(ListView_ValidatercView(plv, &rcView, FALSE)); plv->rcView = rcView; // make sure our scroll positions are correct if (ListView_IsIScrollView(plv)) ListView_FixIScrollPositions(plv, FALSE, &rcClient); ASSERT(ListView_ValidateScrollPositions(plv, &rcClient)); } else { FullRecompute: plv->rcView.left = RECOMPUTE; } } DebugMsg(TF_LISTVIEW, TEXT("LV -- AdjustViewRect pitem %d -- (%x, %x)"), pitem, pitem->pt.x, pitem->pt.y); pitem->pt.x = x; pitem->pt.y = y; // Compute the workarea of this item if applicable ListView_FindWorkArea(plv, pitem->pt, &(pitem->iWorkArea)); } BOOL ListView_OnSetItemPosition(LV* plv, int i, int x, int y) { LISTITEM* pitem; if (plv->fGroupView) return FALSE; if (ListView_IsListView(plv)) return FALSE; if (ListView_IsOwnerData(plv)) { RIPMSG(0, "LVM_SETITEMPOSITION: Invalid for owner-data listview"); return FALSE; } pitem = ListView_GetItemPtr(plv, i); if (!pitem) return FALSE; // // this is a hack to fix a bug in OLE drag/drop loop // if (x >= 0xF000 && x < 0x10000) { DebugMsg(TF_LISTVIEW, TEXT("LV -- On SetItemPosition fixing truncated negative number 0x%08X"), x); x = x - 0x10000; } if (y >= 0xF000 && y < 0x10000) { DebugMsg(TF_LISTVIEW, TEXT("LV -- On SetItemPosition fixing truncated negative number 0x%08X"), y); y = y - 0x10000; } ListView_InvalidateTTLastHit(plv, plv->iTTLastHit); if (pitem->cyFoldedLabel == SRECOMPUTE) { _ListView_RecomputeLabelSize(plv, pitem, i, NULL, FALSE); } // erase old if (y != pitem->pt.y || x != pitem->pt.x) { // Don't invalidate if it hasn't got a position yet if (pitem->pt.y != RECOMPUTE) { ListView_InvalidateItem(plv, i, FALSE, RDW_INVALIDATE | RDW_ERASE); } else if (plv->uUnplaced) { // this means an unplaced item got placed plv->uUnplaced--; if (!plv->uUnplaced) { MSG msg; // if this is now 0, pull out the postmessage PeekMessage(&msg, plv->ci.hwnd, LVMI_PLACEITEMS, LVMI_PLACEITEMS, PM_REMOVE); } } if (y == RECOMPUTE) { // if they're setting the new position to be a "any open spot" post that we // need to calc this later if (!plv->uUnplaced) { PostMessage(plv->ci.hwnd, LVMI_PLACEITEMS, 0, 0); } plv->uUnplaced++; } } DebugMsg(TF_LISTVIEW, TEXT("LV -- On SetItemPosition %d %d %d %d -- (%x, %x)"), plv->rcView.left, plv->rcView.top, plv->rcView.right, plv->rcView.bottom, pitem->pt.x, pitem->pt.y); LV_AdjustViewRectOnMove(plv, pitem, x, y); // and draw at new position ListView_RecalcRegion(plv, FALSE, TRUE); ListView_InvalidateItem(plv, i, FALSE, RDW_INVALIDATE); // If autoarrange is turned on, do it now... if (ListView_RedrawEnabled(plv)) { ListView_ArrangeOrSnapToGrid(plv); if (!(plv->ci.style & LVS_AUTOARRANGE)) ListView_UpdateScrollBars(plv); } if (!(plv->ci.style & LVS_AUTOARRANGE)) { plv->fIconsPositioned = TRUE; } return TRUE; } BOOL ListView_OnGetItemPosition(LV* plv, int i, POINT* ppt) { LISTITEM* pitem; // // This needs to handle all views as it is used to figure out // where the item is during drag and drop and the like // if (!ppt) { RIPMSG(0, "LVM_GETITEMPOSITION: Invalid ppt = NULL"); return FALSE; } if (ListView_IsListView(plv) || ListView_IsReportView(plv) || ListView_IsOwnerData(plv)) { RECT rcIcon; ListView_GetRects(plv, i, QUERY_DEFAULT, &rcIcon, NULL, NULL, NULL); ppt->x = rcIcon.left; ppt->y = rcIcon.top; } else { pitem = ListView_GetItemPtr(plv, i); if (!pitem) return FALSE; if (pitem->pt.x == RECOMPUTE) ListView_Recompute(plv); ppt->x = pitem->pt.x; ppt->y = pitem->pt.y; } return TRUE; } BOOL ListView_OnGetOrigin(LV* plv, POINT* ppt) { if (!ppt) { DebugMsg(DM_ERROR, TEXT("ListView_OnGetOrigin: ppt is NULL")); return FALSE; } if (ListView_IsListView(plv) || ListView_IsReportView(plv)) return FALSE; *ppt = plv->ptOrigin; return TRUE; } int ListView_OnGetStringWidthA(LV* plv, LPCSTR psz, HDC hdc) { LPWSTR pszW = NULL; int iRet; if (!psz) return 0; if ((psz != NULL) && (pszW = ProduceWFromA(plv->ci.uiCodePage, psz)) == NULL) return 0; iRet = ListView_OnGetStringWidth(plv, pszW, hdc); FreeProducedString(pszW); return iRet; } int ListView_OnGetStringWidth(LV* plv, LPCTSTR psz, HDC hdc) { SIZE siz; HDC hdcFree = NULL; HFONT hfontPrev; if (!psz || psz == LPSTR_TEXTCALLBACK) return 0; if (!hdc) { hdcFree = hdc = GetDC(plv->ci.hwnd); } hfontPrev = SelectFont(hdc, plv->hfontLabel); GetTextExtentPoint(hdc, psz, lstrlen(psz), &siz); SelectFont(hdc, hfontPrev); if (hdcFree) { ReleaseDC(plv->ci.hwnd, hdcFree); } return siz.cx; } int ListView_OnGetColumnWidth(LV* plv, int iCol) { if (ListView_IsReportView(plv)) return ListView_RGetColumnWidth(plv, iCol); else if (ListView_IsListView(plv)) return plv->cxItem; return 0; } BOOL ListView_ISetColumnWidth(LV* plv, int iCol, int cx, BOOL fExplicit) { if (ListView_IsListView(plv)) { if (iCol != 0 || cx <= 0) return FALSE; // if it's different and this is an explicit set, or we've never set it explicitly if (plv->cxItem != cx && (fExplicit || !(plv->flags & LVF_COLSIZESET))) { // REVIEW: Should optimize what gets invalidated here... plv->cxItem = cx; if (fExplicit) plv->flags |= LVF_COLSIZESET; // Set the fact that we explictly set size!. if (ListView_IsLabelTip(plv)) { // A truncated label may have been exposed or vice versa. ListView_InvalidateTTLastHit(plv, plv->iTTLastHit); } RedrawWindow(plv->ci.hwnd, NULL, NULL, RDW_INVALIDATE | RDW_ERASE); ListView_UpdateScrollBars(plv); } return TRUE; } else if (ListView_IsReportView(plv)) { if (ListView_IsLabelTip(plv)) { // A truncated label may have been exposed or vice versa. ListView_InvalidateTTLastHit(plv, plv->iTTLastHit); } return ListView_RSetColumnWidth(plv, iCol, cx); } else { if (cx && plv->cxItem != cx && (fExplicit || !(plv->flags & LVF_COLSIZESET))) { // REVIEW: Should optimize what gets invalidated here... plv->cxItem = cx; if (fExplicit) plv->flags |= LVF_COLSIZESET; // Set the fact that we explictly set size!. RedrawWindow(plv->ci.hwnd, NULL, NULL, RDW_INVALIDATE | RDW_ERASE); ListView_UpdateScrollBars(plv); } // BUG-FOR-BUG COMPATIBILITY: IE4 accidentally returned FALSE here. } return FALSE; } void DrawGradiantLine(HDC hdc, RECT* prcText, RECT* prcGroup) { COLORREF cr1 = GetSysColor(COLOR_GRADIENTACTIVECAPTION); COLORREF cr2 = GetSysColor(COLOR_WINDOW); TRIVERTEX pt[2]; GRADIENT_RECT gr; RECT rc = {prcGroup->left, prcText->bottom-1, prcGroup->left + GRADIENT_WIDTH, prcText->bottom}; pt[0].x = rc.left; pt[0].y = rc.top; pt[1].x = rc.right; pt[1].y = rc.bottom; pt[0].Red = GetRValue(cr1) << 8; pt[0].Green = GetGValue(cr1) << 8; pt[0].Blue = GetBValue(cr1) << 8; pt[0].Alpha = 0xFF00; pt[1].Red = GetRValue(cr2) << 8; pt[1].Green = GetGValue(cr2) << 8; pt[1].Blue = GetBValue(cr2) << 8; pt[1].Alpha = 0x0000; gr.UpperLeft = 0; gr.LowerRight = 1; GdiGradientFill(hdc, pt, 2, &gr, 1, GRADIENT_FILL_RECT_H); } void ListView_Redraw(LV* plv, HDC hdc, RECT* prcClip) { int i = 0; int cItem = ListView_Count(plv); NMCUSTOMDRAW nmcd; LVDRAWITEM lvdi = {0}; SetBkMode(hdc, TRANSPARENT); SelectFont(hdc, plv->hfontLabel); nmcd.hdc = hdc; nmcd.rc = *prcClip; plv->ci.dwCustom = CICustomDrawNotify(&plv->ci, CDDS_PREPAINT, &nmcd); if (!(plv->ci.dwCustom & CDRF_SKIPDEFAULT)) { int cGroups; // Just before doing any painting, see if the region is up to date... ListView_RecalcRegion(plv, FALSE, TRUE); // // For list view and report view, we can save a lot of time // by calculating the index of the first item that may need // painting... // switch (plv->wView) { case LV_VIEW_DETAILS: if (!plv->fGroupView) { i = ListView_RYHitTest(plv, prcClip->top); cItem = ListView_RYHitTest(plv, prcClip->bottom) + 1; } break; case LV_VIEW_LIST: i = ListView_LCalcViewItem(plv, prcClip->left, prcClip->top); cItem = ListView_LCalcViewItem(plv, prcClip->right, prcClip->bottom) + 1; break; default: if (ListView_IsOwnerData(plv)) { ListView_CalcMinMaxIndex(plv, prcClip, &i, &cItem); break; } } if (i < 0) i = 0; cItem = min(ListView_Count(plv), cItem); if (ListView_IsOwnerData(plv) && (cItem > i)) { ListView_NotifyCacheHint(plv, i, cItem-1); ListView_LazyCreateWinEvents(plv, i, cItem-1); } lvdi.plv = plv; lvdi.nmcd.nmcd.hdc = hdc; lvdi.prcClip = prcClip; lvdi.pitem = NULL; if (plv->hdpaGroups) { cGroups = DPA_GetPtrCount(plv->hdpaGroups); if (plv->fGroupView && cGroups > 0 && ListView_IsGroupedView(plv)) { int iGroup; RECT rcClient; GetClientRect(plv->ci.hwnd, &rcClient); for (iGroup = 0; iGroup < cGroups; iGroup++) { LISTGROUP* pgrp = DPA_FastGetPtr(plv->hdpaGroups, iGroup); int cItems = DPA_GetPtrCount(pgrp->hdpa); if (cItems > 0) { RECT rcT; RECT rc; SetRect(&rc, 0, pgrp->rc.top - LISTGROUP_HEIGHT(plv, pgrp), rcClient.right, pgrp->rc.bottom + plv->rcBorder.bottom + plv->paddingBottom); if (ListView_IsReportView(plv)) { OffsetRect(&rc, -plv->ptlRptOrigin.x, -plv->ptlRptOrigin.y + plv->yTop); } else { OffsetRect(&rc, -plv->ptOrigin.x, -plv->ptOrigin.y); } if (IntersectRect(&rcT, &rc, prcClip)) { NMLVCUSTOMDRAW nmcdGroup = {0}; DWORD dwCust; UINT uAlign = LVCFMT_LEFT; HFONT hfontOld; RECT rcBorder = plv->rcBorder; rcBorder.top = max(pgrp->cyTitle + 6, plv->rcBorder.top); nmcdGroup.nmcd.hdc = hdc; nmcdGroup.nmcd.rc = rc; nmcdGroup.nmcd.dwItemSpec = pgrp->iGroupId; nmcdGroup.dwItemType = LVCDI_GROUP; nmcdGroup.rcText.left = rc.left + plv->paddingLeft; nmcdGroup.rcText.top = rc.top; nmcdGroup.rcText.bottom = rc.top + max(pgrp->cyTitle + 6, plv->rcBorder.top); nmcdGroup.rcText.right = rc.right; nmcdGroup.uAlign = pgrp->uAlign; nmcdGroup.clrText = plv->crHeader; dwCust = CICustomDrawNotify(&plv->ci, CDDS_PREPAINT, &nmcdGroup.nmcd); if (!(dwCust & CDRF_SKIPDEFAULT)) { RECT rcHeader = {0}; if (!(LVCDRF_NOGROUPFRAME & dwCust)) { DrawGradiantLine(hdc, &nmcdGroup.rcText, &nmcdGroup.nmcd.rc); } if (!(dwCust & CDRF_NEWFONT)) { hfontOld = SelectObject(hdc, plv->hfontGroup); } if (nmcdGroup.uAlign & LVGA_HEADER_CENTER) uAlign = LVCFMT_CENTER; else if (nmcdGroup.uAlign & LVGA_HEADER_RIGHT) uAlign = LVCFMT_RIGHT; SHDrawText(hdc, pgrp->pszHeader, &nmcdGroup.rcText, uAlign, SHDT_VCENTER | SHDT_LEFT, plv->cyLabelChar, plv->cxEllipses, nmcdGroup.clrText, CLR_NONE); // Need do do this before we unselect so that we get the right font... DrawText(hdc, pgrp->pszHeader, -1, &rcHeader, DT_LV | DT_CALCRECT); if (!(dwCust & CDRF_NEWFONT)) { SelectObject(hdc, hfontOld); } } dwCust = CICustomDrawNotify(&plv->ci, CDDS_POSTPAINT, &nmcdGroup.nmcd); } } } } } cItem = min(ListView_Count(plv), cItem); for (; i < cItem; i++) { BOOL bSuccess; int i2; if (ListView_IsRearrangeableView(plv) && !ListView_IsOwnerData(plv)) { LISTITEM *pitem; // Icon views: Draw back-to-front mapped through // Z-order array for proper Z order appearance - If autoarrange // is on, we don't need to do this as our arrange code is setup // to not overlap items! // // For the cases where we might have overlap, we sped this up, // by converting the hdpaZorder into a list of indexes instead // of pointers. This ovoids the costly convert pointer to // index call. // i2 = (int)(UINT_PTR)DPA_FastGetPtr(plv->hdpaZOrder, (cItem - 1) -i); // // do a fast clip check on the item so we dont even try to // draw it unless it is visible // // for small icon view we cant clip on the left without // getting the text // // for large icon view we cant clip on the top without // getting the text // // for large icon view in NOLABELWRAP mode, we can't clip // on the top without getting the text, nor can we clip to // the left or right in case the text is long. // // we can always clip to the bottom // pitem = ListView_FastGetItemPtr(plv, i2); if (pitem && pitem->pt.x != RECOMPUTE) { int yBias = 0; if (ListView_IsBorderSelect(plv)) yBias = BORDERSELECT_THICKNESS; if (pitem->pt.y - yBias - plv->ptOrigin.y > prcClip->bottom) continue; if (plv->wView == LV_VIEW_SMALLICON) { if (pitem->pt.x - plv->ptOrigin.x - plv->cxState > prcClip->right) continue; if (pitem->pt.y + yBias + plv->cyItem - plv->ptOrigin.y < prcClip->top) continue; } else if (!(plv->ci.style & LVS_NOLABELWRAP)) { if (plv->wView == LV_VIEW_TILE) { if (pitem->pt.x - plv->sizeTile.cx - plv->ptOrigin.x > prcClip->right) continue; if (pitem->pt.x + yBias + plv->sizeTile.cx - plv->ptOrigin.x < prcClip->left) continue; } else // LV_VIEW_ICON { if (pitem->pt.x - plv->cxIconSpacing - plv->ptOrigin.x > prcClip->right) continue; if (pitem->pt.x + yBias + plv->cxIconSpacing - plv->ptOrigin.x < prcClip->left) continue; } } } if (plv->fGroupView && !LISTITEM_HASGROUP(pitem)) { continue; // Don't paint items not in a group. } } else i2 = i; plv->iItemDrawing = i2; lvdi.nmcd.nmcd.dwItemSpec = i2; // these may get changed lvdi.lpptOrg = NULL; lvdi.flags = 0; lvdi.nmcd.clrText = plv->clrText; lvdi.nmcd.clrTextBk = plv->clrTextBk; lvdi.nmcd.clrFace = plv->clrBk; lvdi.nmcd.iIconEffect = ILD_NORMAL; lvdi.nmcd.iIconPhase = 0; bSuccess = ListView_DrawItem(&lvdi); if (!bSuccess) { break; } } if (ListView_IsRearrangeableView(plv) && (ListView_IsOwnerData(plv)) && plv->iFocus != -1) { // since there's no zorder in ownerdata, we explicitly draw the focus guy last (again) // so that it'll appear on top // we may potentially want to do this for all items that are selected plv->iItemDrawing = plv->iFocus; lvdi.nmcd.nmcd.dwItemSpec = plv->iItemDrawing; // these may get changed lvdi.lpptOrg = NULL; lvdi.flags = 0; lvdi.nmcd.clrText = plv->clrText; lvdi.nmcd.clrTextBk = plv->clrTextBk; ListView_DrawItem(&lvdi); } // this is an NT5/Memphis feature. if (ListView_Count(plv) == 0) { // there're no items in this view // check if we need to display some text in this case. if (ListView_GetEmptyText(plv)) { RECT rcClip; UINT flags = 0; // Put some edging between the text and the border of the // window so we don't slam up against the border. // This keeps DBCS from looking horrid. rcClip.left = g_cxEdge; rcClip.top = g_cyEdge; if (plv->dwExStyle & WS_EX_RTLREADING) flags |= SHDT_RTLREADING; // if its a report view && we have a header then move the text down if (ListView_IsReportView(plv) && (!(plv->ci.style & LVS_NOCOLUMNHEADER))) rcClip.top += plv->cyItem; // Note: Use the full sizeClient.cx as the right margin // in case pszEmptyText is wider than the client rectangle. rcClip.left -= (int)plv->ptlRptOrigin.x; rcClip.right = plv->sizeClient.cx; rcClip.bottom = rcClip.top + plv->cyItem; SHDrawText(hdc, plv->pszEmptyText, &rcClip, LVCFMT_LEFT, flags, plv->cyLabelChar, plv->cxEllipses, plv->clrText, plv->clrBk); } } plv->iItemDrawing = -1; // post painting.... this is to do any extra (non item) painting // such a grid lines switch (plv->wView) { case LV_VIEW_DETAILS: ListView_RAfterRedraw(plv, hdc); break; } // Insert mark { RECT rcInsertMark; if (ListView_OnGetInsertMarkRect(plv, &rcInsertMark)) { OffsetRect(&rcInsertMark, -plv->ptOrigin.x, -plv->ptOrigin.y); CCDrawInsertMark(hdc, &rcInsertMark, ((plv->ci.style & LVS_ALIGNMASK) == LVS_ALIGNTOP), ListView_OnGetInsertMarkColor(plv)); } } // notify parent afterwards if they want us to if (plv->ci.dwCustom & CDRF_NOTIFYPOSTPAINT) { CICustomDrawNotify(&plv->ci, CDDS_POSTPAINT, &nmcd); } } } BOOL ListView_DrawItem(PLVDRAWITEM plvdi) { BOOL fAllowHotSelection = FALSE; BOOL bRet = TRUE; UINT state; if (!ListView_IsOwnerData(plvdi->plv) && (!plvdi->plv->hdpa || plvdi->nmcd.nmcd.dwItemSpec > (UINT)DPA_GetPtrCount(plvdi->plv->hdpa))) return FALSE; if (!ListView_IsOwnerData(plvdi->plv)) { plvdi->pitem = ListView_FastGetItemPtr(plvdi->plv, plvdi->nmcd.nmcd.dwItemSpec); } // notify on custom draw then do it! plvdi->nmcd.nmcd.uItemState = 0; plvdi->nmcd.nmcd.lItemlParam = (plvdi->pitem)? plvdi->pitem->lParam : 0; if (!(plvdi->flags & LVDI_NOWAYFOCUS)) { if (plvdi->plv->flags & LVF_FOCUSED) { // if we're ownerdraw or asked to callback, go // fetch the state if (!plvdi->pitem || (plvdi->plv->stateCallbackMask & (LVIS_SELECTED | LVIS_FOCUSED))) { state = (WORD) ListView_OnGetItemState(plvdi->plv, (int) plvdi->nmcd.nmcd.dwItemSpec, LVIS_SELECTED | LVIS_FOCUSED); } else { state = plvdi->pitem->state; } if (state & LVIS_FOCUSED) { plvdi->nmcd.nmcd.uItemState |= CDIS_FOCUS; } if (state & LVIS_SELECTED) { plvdi->nmcd.nmcd.uItemState |= CDIS_SELECTED; } } // NOTE: This is a bug. We should set CDIS_SELECTED only if the item // really is selected. But this bug has existed forever so who knows // what apps are relying on it. Standard workaround is for the client // to do a GetItemState and reconfirm the LVIS_SELECTED flag. // That's what we do in ListView_DrawImageEx. if (plvdi->plv->ci.style & LVS_SHOWSELALWAYS) { plvdi->nmcd.nmcd.uItemState |= CDIS_SELECTED; } } if (!(CCGetUIState(&(plvdi->plv->ci)) & UISF_HIDEFOCUS)) { plvdi->nmcd.nmcd.uItemState |= CDIS_SHOWKEYBOARDCUES; } plvdi->nmcd.clrText = plvdi->plv->clrText; plvdi->nmcd.clrTextBk = (plvdi->plv->ci.style & WS_DISABLED ? plvdi->plv->clrBk : plvdi->plv->clrTextBk); if ((plvdi->plv->exStyle & LVS_EX_UNDERLINEHOT) && plvdi->plv->iHot == (int)plvdi->nmcd.nmcd.dwItemSpec && (plvdi->plv->exStyle & LVS_EX_ONECLICKACTIVATE) || ((plvdi->plv->exStyle & LVS_EX_TWOCLICKACTIVATE) && ListView_OnGetItemState(plvdi->plv, (int) plvdi->nmcd.nmcd.dwItemSpec, LVIS_SELECTED))) { fAllowHotSelection = TRUE; // Handle the HOT case if (plvdi->plv->clrHotlight != CLR_DEFAULT) { plvdi->nmcd.clrText = plvdi->plv->clrHotlight; } else { plvdi->nmcd.clrText = GetSysColor(COLOR_HOTLIGHT); } // if hotlight color is the same as the background // color you don't see the text -- slam to a visible color in this case. if (plvdi->nmcd.clrText == plvdi->nmcd.clrTextBk) { if (COLORISLIGHT(plvdi->nmcd.clrTextBk)) plvdi->nmcd.clrText = 0x000000; // black else plvdi->nmcd.clrText = 0xFFFFFF; // white } SelectFont(plvdi->nmcd.nmcd.hdc, plvdi->plv->hFontHot); plvdi->nmcd.nmcd.uItemState |= CDIS_HOT; } else if ((plvdi->plv->exStyle & LVS_EX_ONECLICKACTIVATE) || ((plvdi->plv->exStyle & LVS_EX_TWOCLICKACTIVATE) && ListView_OnGetItemState(plvdi->plv, (int) plvdi->nmcd.nmcd.dwItemSpec, LVIS_SELECTED))) { // Handle the non-hot webview case if ((plvdi->plv->exStyle & LVS_EX_UNDERLINECOLD) && plvdi->plv->hFontHot) { SelectFont(plvdi->nmcd.nmcd.hdc, plvdi->plv->hFontHot); } else { SelectFont(plvdi->nmcd.nmcd.hdc, plvdi->plv->hfontLabel); } } else { // Handle the non-webview case SelectFont(plvdi->nmcd.nmcd.hdc, plvdi->plv->hfontLabel); } plvdi->dwCustom = CICustomDrawNotify(&plvdi->plv->ci, CDDS_ITEMPREPAINT, &plvdi->nmcd.nmcd); plvdi->flags &= ~(LVDI_FOCUS | LVDI_SELECTED); if (plvdi->nmcd.nmcd.uItemState & CDIS_FOCUS) plvdi->flags |= LVDI_FOCUS; if (plvdi->nmcd.nmcd.uItemState & CDIS_SELECTED) { if (plvdi->plv->flags & LVF_FOCUSED) plvdi->flags |= LVDI_SELECTED; else plvdi->flags |= LVDI_SELECTNOFOCUS; if (plvdi->plv->iHot == (int)plvdi->nmcd.nmcd.dwItemSpec && fAllowHotSelection) plvdi->flags |= LVDI_HOTSELECTED; } if (!(plvdi->dwCustom & CDRF_SKIPDEFAULT)) { if (!ListView_IsOwnerData(plvdi->plv)) { if (plvdi->dwCustom & CDRF_NEWFONT) { _ListView_RecomputeLabelSize(plvdi->plv, plvdi->pitem, (int) plvdi->nmcd.nmcd.dwItemSpec, plvdi->nmcd.nmcd.hdc, FALSE); } } bRet = _ListView_DrawItem(plvdi); if (plvdi->dwCustom & CDRF_NOTIFYPOSTPAINT) { plvdi->nmcd.iSubItem = 0; CICustomDrawNotify(&plvdi->plv->ci, CDDS_ITEMPOSTPAINT, &plvdi->nmcd.nmcd); } if (plvdi->dwCustom & CDRF_NEWFONT) { SelectObject(plvdi->nmcd.nmcd.hdc, plvdi->plv->hfontLabel); plvdi->plv->flags |= LVF_CUSTOMFONT; } } return bRet; } void WINAPI SHThemeDrawText(HTHEME hTheme, HDC hdc, int iPartId, int iStateId, LPCTSTR pszText, RECT* prc, int fmt, UINT flags, int cyChar, int cxEllipses, COLORREF clrText, COLORREF clrTextBk) { int cchText; COLORREF clrSave = GetTextColor(hdc), clrSaveBk = 0; RECT rc; UINT uETOFlags = 0; BOOL fForeOnly = FALSE; TCHAR ach[CCHLABELMAX + CCHELLIPSES]; int align; BOOL fUseShadowedText = (flags & SHDT_SHADOWTEXT) && (!g_fHighContrast); if (!pszText) return; if (IsRectEmpty(prc)) return; if (flags & SHDT_RTLREADING) { align = GetTextAlign(hdc); SetTextAlign(hdc, align | TA_RTLREADING); } rc = *prc; if (fUseShadowedText) { if (!AreAllMonitorsAtLeast(16)) fUseShadowedText = FALSE; } // If needed, add in a little extra margin... // if (flags & SHDT_EXTRAMARGIN) { rc.left += g_cxLabelMargin * 3; rc.right -= g_cxLabelMargin * 3; } else if (!(flags & SHDT_NOMARGIN)) { rc.left += g_cxLabelMargin; rc.right -= g_cxLabelMargin; } if ((rc.left >= rc.right) && !(flags & (SHDT_SELECTED | SHDT_DESELECTED | SHDT_SELECTNOFOCUS))) return; if ((flags & SHDT_ELLIPSES) && ListView_NeedsEllipses(hdc, pszText, &rc, &cchText, cxEllipses)) { // In some cases cchText was comming back bigger than // ARRYASIZE(ach), so we need to make sure we don't overflow the buffer // if cchText is too big for the buffer, truncate it down to size if (cchText >= ARRAYSIZE(ach) - CCHELLIPSES) cchText = ARRAYSIZE(ach) - CCHELLIPSES - 1; memcpy(ach, pszText, cchText * sizeof(TCHAR)); lstrcpy(ach + cchText, c_szEllipses); pszText = ach; // Left-justify, in case there's no room for all of ellipses // fmt = LVCFMT_LEFT; cchText += CCHELLIPSES; } else { cchText = lstrlen(pszText); } if (((clrTextBk == CLR_NONE) && !(flags & (SHDT_SELECTED | SHDT_SELECTNOFOCUS))) || (flags & SHDT_TRANSPARENT)) { fForeOnly = TRUE; clrSave = SetTextColor(hdc, (flags & SHDT_TRANSPARENT) ? 0 : clrText); } else if (!hTheme || clrTextBk != CLR_NONE) { HBRUSH hbrUse = NULL; HBRUSH hbrDelete = NULL; uETOFlags |= ETO_OPAQUE; if ((flags & SHDT_SELECTED || flags & SHDT_SELECTNOFOCUS) && !(flags & SHDT_NOSELECTED)) { fUseShadowedText = FALSE; if (flags & SHDT_SELECTNOFOCUS) { clrText = g_clrBtnText; clrTextBk = g_clrBtnFace; if (flags & SHDT_DRAWTEXT) { hbrUse = g_hbrBtnFace; } } else { clrText = g_clrHighlightText; if (flags & SHDT_HOTSELECTED) clrTextBk = GetSysColor(COLOR_HOTLIGHT); else clrTextBk = g_clrHighlight; if (flags & SHDT_DRAWTEXT) hbrUse = (flags & SHDT_HOTSELECTED)?GetSysColorBrush(COLOR_HOTLIGHT): g_hbrHighlight; } } else if (clrText == CLR_DEFAULT && clrTextBk == CLR_DEFAULT) { fUseShadowedText = FALSE; clrText = g_clrWindowText; clrTextBk = g_clrWindow; if ((flags & (SHDT_DRAWTEXT | SHDT_DESELECTED)) == (SHDT_DRAWTEXT | SHDT_DESELECTED)) { hbrUse = g_hbrWindow; } } else { if (clrText == CLR_DEFAULT) clrText = g_clrWindowText; if (clrTextBk == CLR_DEFAULT) clrTextBk = g_clrWindow; if (fUseShadowedText == FALSE && ((flags & (SHDT_DRAWTEXT | SHDT_DESELECTED)) == (SHDT_DRAWTEXT | SHDT_DESELECTED) || hTheme)) { hbrUse = CreateSolidBrush(GetNearestColor(hdc, clrTextBk)); if (hbrUse) { hbrDelete = hbrUse; } else hbrUse = GetStockObject(WHITE_BRUSH); } } // now set it clrSave = SetTextColor(hdc, clrText); clrSaveBk = SetBkColor(hdc, clrTextBk); if (hbrUse) { FillRect(hdc, prc, hbrUse); if (hbrDelete) DeleteObject(hbrDelete); } } // If we want the item to display as if it was depressed, we will // offset the text rectangle down and to the left if (flags & SHDT_DEPRESSED) OffsetRect(&rc, g_cxBorder, g_cyBorder); if (flags & SHDT_DRAWTEXT || hTheme) { HRESULT hr = E_FAIL; UINT uDTFlags; if (flags & SHDT_DRAWTEXT) { uDTFlags= DT_LVWRAP | DT_END_ELLIPSIS; } else { uDTFlags = DT_END_ELLIPSIS | DT_SINGLELINE | DT_VCENTER; if (fmt & LVCFMT_CENTER) uDTFlags |= DT_CENTER; else if (fmt & LVCFMT_RIGHT) uDTFlags |= DT_RIGHT; } if (flags & SHDT_DTELLIPSIS) uDTFlags |= DT_WORD_ELLIPSIS; if (!(flags & SHDT_CLIPPED)) uDTFlags |= DT_NOCLIP; if (flags & SHDT_NODBCSBREAK) uDTFlags |= DT_NOFULLWIDTHCHARBREAK; if (flags & SHDT_VCENTER) uDTFlags |= DT_VCENTER | DT_SINGLELINE; if (flags & SHDT_LEFT) uDTFlags = DT_LEFT | uDTFlags & ~DT_CENTER; if (fUseShadowedText) { DrawShadowText(hdc, pszText, cchText, &rc, uDTFlags, RGB(255,255,255), RGB(0,0,0), 2, 2); } else { if (hTheme) hr = DrawThemeText(hTheme, hdc, iPartId, iStateId, pszText, -1, uDTFlags, 0, &rc); if (FAILED(hr)) DrawText(hdc, pszText, cchText, &rc, uDTFlags); } } else { if (fmt != LVCFMT_LEFT) { SIZE siz; GetTextExtentPoint(hdc, pszText, cchText, &siz); if (fmt == LVCFMT_CENTER) rc.left = (rc.left + rc.right - siz.cx) / 2; else // fmt == LVCFMT_RIGHT rc.left = rc.right - siz.cx; } // Center vertically in case the bitmap (to the left) is larger than // the height of one line rc.top += (rc.bottom - rc.top - cyChar) / 2; if (flags & SHDT_CLIPPED) uETOFlags |= ETO_CLIPPED; // HACK: ExtTextOut() has an off-by-one bug in its rendering of RTL // text. We need this hack to render properly (RAID 439915). // Note that this bug is NOT present in the DrawText() API. if (flags & SHDT_RTLREADING) rc.left--; ExtTextOut(hdc, rc.left, rc.top, uETOFlags, prc, pszText, cchText, NULL); } if (flags & (SHDT_SELECTED | SHDT_DESELECTED | SHDT_TRANSPARENT)) { SetTextColor(hdc, clrSave); if (!fForeOnly) SetBkColor(hdc, clrSaveBk); } if (flags & SHDT_RTLREADING) SetTextAlign(hdc, align); } void WINAPI SHDrawText(HDC hdc, LPCTSTR pszText, RECT* prc, int fmt, UINT flags, int cyChar, int cxEllipses, COLORREF clrText, COLORREF clrTextBk) { SHThemeDrawText(NULL, hdc, 0, 0, pszText, prc, fmt, flags, cyChar, cxEllipses, clrText, clrTextBk); } /*---------------------------------------------------------------- ** Create an imagelist to be used for dragging. ** ** 1) create mask and image bitmap matching the select bounds size ** 2) draw the text to both bitmaps (in black for now) ** 3) create an imagelist with these bitmaps ** 4) make a dithered copy of the image onto the new imagelist **----------------------------------------------------------------*/ HIMAGELIST ListView_OnCreateDragImage(LV *plv, int iItem, LPPOINT lpptUpLeft) { HWND hwndLV = plv->ci.hwnd; RECT rcBounds, rcImage, rcLabel; HDC hdcMem = NULL; HBITMAP hbmImage = NULL; HBITMAP hbmMask = NULL; HBITMAP hbmOld; HIMAGELIST himl = NULL; int dx, dy; HIMAGELIST himlSrc; LV_ITEM item; POINT ptOrg; LVDRAWITEM lvdi = {0}; RECT rcSelBounds; BOOL bMirroredWnd = (plv->ci.dwExStyle&RTL_MIRRORED_WINDOW); int iImageList; if (!lpptUpLeft) return NULL; if (iItem >= ListView_Count(plv)) return NULL; if (plv->iHot == iItem) { ListView_OnSetHotItem(plv, -1); UpdateWindow(plv->ci.hwnd); } ListView_GetRects(plv, iItem, QUERY_DEFAULT, &rcImage, &rcLabel, &rcBounds, &rcSelBounds); if (ListView_IsIconView(plv)) { ListView_UnfoldRects(plv, iItem, &rcImage, &rcLabel, &rcBounds, &rcSelBounds); InflateRect(&rcImage, -g_cxIconMargin, -g_cyIconMargin); } // chop off any extra filler above icon ptOrg.x = rcBounds.left - rcSelBounds.left; ptOrg.y = rcBounds.top - rcImage.top; dx = rcSelBounds.right - rcSelBounds.left; dy = rcSelBounds.bottom - rcImage.top; lpptUpLeft->x = rcSelBounds.left; lpptUpLeft->y = rcImage.top; if (!(hdcMem = CreateCompatibleDC(NULL))) goto CDI_Exit; if (!(hbmImage = CreateColorBitmap(dx, dy))) goto CDI_Exit; if (!(hbmMask = CreateMonoBitmap(dx, dy))) goto CDI_Exit; // // Mirror the memory DC so that the transition from // mirrored(memDC)->non-mirrored(imagelist DCs)->mirrored(screenDC) // is consistent. [samera] // if (bMirroredWnd) { SET_DC_RTL_MIRRORED(hdcMem); } // prepare for drawing the item SelectObject(hdcMem, plv->hfontLabel); SetBkMode(hdcMem, TRANSPARENT); lvdi.plv = plv; lvdi.nmcd.nmcd.dwItemSpec = iItem; lvdi.pitem = NULL; // make sure it is null as Owner data uses this to trigger things... lvdi.nmcd.nmcd.hdc = hdcMem; lvdi.lpptOrg = &ptOrg; lvdi.prcClip = NULL; lvdi.flags = LVDI_NOIMAGE | LVDI_TRANSTEXT | LVDI_NOWAYFOCUS | LVDI_UNFOLDED; lvdi.nmcd.clrFace = CLR_NONE; /* ** draw the text to both bitmaps */ hbmOld = SelectObject(hdcMem, hbmImage); // fill image with black for transparency PatBlt(hdcMem, 0, 0, dx, dy, BLACKNESS); ListView_DrawItem(&lvdi); if (bMirroredWnd) MirrorBitmapInDC(hdcMem, hbmImage); lvdi.flags = LVDI_NOIMAGE | LVDI_TRANSTEXT | LVDI_NOWAYFOCUS | LVDI_UNFOLDED; SelectObject(hdcMem, hbmMask); // fill mask with white for transparency PatBlt(hdcMem, 0, 0, dx, dy, WHITENESS); ListView_DrawItem(&lvdi); if (bMirroredWnd) MirrorBitmapInDC(hdcMem, hbmMask); // unselect objects that we used SelectObject(hdcMem, hbmOld); SelectObject(hdcMem, g_hfontSystem); if (ListView_IsIconView(plv) || ListView_IsTileView(plv)) iImageList = LVSIL_NORMAL; else iImageList = LVSIL_SMALL; himlSrc = ListView_OnGetImageList(plv, iImageList); /* ** make an image list that for now only has the text ** we use ImageList_Clone so we get a imagelist that ** the same color depth as our own imagelist */ if (!(himl = ImageList_Clone(himlSrc, dx, dy, ILC_MASK, 1, 0))) goto CDI_Exit; ImageList_SetBkColor(himl, CLR_NONE); ImageList_Add(himl, hbmImage, hbmMask); /* ** make a dithered copy of the image part onto our bitmaps ** (need both bitmap and mask to be dithered) */ if (himlSrc) { item.iItem = iItem; item.iSubItem = 0; item.mask = LVIF_IMAGE |LVIF_STATE; item.stateMask = LVIS_OVERLAYMASK; ListView_OnGetItem(plv, &item); ImageList_CopyDitherImage(himl, 0, rcImage.left - rcSelBounds.left, 0, himlSrc, item.iImage, ((plv->ci.dwExStyle & dwExStyleRTLMirrorWnd) ? ILD_MIRROR : 0L) | (item.state & LVIS_OVERLAYMASK)); } CDI_Exit: if (hdcMem) DeleteObject(hdcMem); if (hbmImage) DeleteObject(hbmImage); if (hbmMask) DeleteObject(hbmMask); return himl; } // ListView_OnGetTopIndex -- Gets the index of the first visible item // For list view and report view this calculates the actual index // for iconic views it alway returns 0 // int ListView_OnGetTopIndex(LV* plv) { if (ListView_IsReportView(plv) && !plv->fGroupView) return (int)((plv->ptlRptOrigin.y) / plv->cyItem); else if (ListView_IsListView(plv)) return (plv->xOrigin / plv->cxItem) * plv->cItemCol; else return 0; } // ListView_OnGetCountPerPage -- Gets the count of items that will fit // on a page For list view and report view this calculates the // count depending on the size of the window and for Iconic views it // will always return the count of items in the list view. // int ListView_OnGetCountPerPage(LV* plv) { if (ListView_IsReportView(plv)) return (plv->sizeClient.cy - plv->yTop) / plv->cyItem; else if (ListView_IsListView(plv)) return ((plv->sizeClient.cx)/ plv->cxItem) * plv->cItemCol; else return (ListView_Count(plv)); } /* Purpose: / Provides support for invalidating items within list views. / / Notes: / Copes with invalidating the extra region in the list view that requires / us to erase the background. Design to optimise out the ERASURE of the / background. / / For details on the API see ListView_InvalidateItem. / / In: / plv->ListView structure to work with / iItem = item number / bSrelectionOnly = refesh the selection / fRedraw = Flags for RedrawWindow / Out: / - */ void ListView_InvalidateFoldedItem(LV* plv, int iItem, BOOL fSelectionOnly, UINT fRedraw) { ListView_InvalidateItem(plv, iItem, fSelectionOnly, fRedraw); if (ListView_IsIconView(plv) && (!ListView_IsItemUnfolded(plv, iItem) || (fRedraw & RDW_ERASE))) { RECT rcLabel; if (ListView_GetUnfoldedRect(plv, iItem, &rcLabel)) { RedrawWindow(plv->ci.hwnd, &rcLabel, NULL, fRedraw|RDW_ERASE); } } } /* / Purpose: / Having previously called get rects, then call this function to ensure / that they are correctly unfolded. / / Notes: / - / / In: / plv-> list view to unfold on / iItem = item number / prcIcon -> icon bounding box / prcLabel -> rectangle for the label structure / prcBounds -> bounds rectangle / == NULL for none / These are currently the same for large icons / prcSelectBounds -> selection bounds / == NULL / / Out: TRUE if unfolding the item was worth anything / - */ BOOL ListView_UnfoldRects(LV* plv, int iItem, RECT * prcIcon, RECT * prcLabel, RECT * prcBounds, RECT * prcSelectBounds) { LISTITEM item; LISTITEM* pitem = &item; BOOL fRc = FALSE; if (!ListView_IsIconView(plv)) return fRc; // If we have a label pointer then expand as required // nb - different paths for owner data if (prcLabel) { if (!ListView_IsOwnerData(plv)) { pitem = ListView_GetItemPtr(plv, iItem); if (!EVAL(pitem)) { // DavidShi was able to get us into here with an invalid // item number during a delete notification. So if the // item number is invalid, just return a blank rectangle // instead of faulting. SetRectEmpty(prcLabel); goto doneLabel; } } else { _ListView_RecomputeLabelSize(plv, pitem, iItem, NULL, FALSE); } if (prcLabel->bottom != prcLabel->top + max(pitem->cyUnfoldedLabel, pitem->cyFoldedLabel)) fRc = TRUE; // In HideLabel mode it's always "worthwhile" to "unfold" the rects because the label is not shown // by default. By returning TRUE we cause the item's label to display in a tooltip where the label // would normally be. if (ListView_HideLabels(plv)) fRc = TRUE; prcLabel->bottom = prcLabel->top + pitem->cyUnfoldedLabel; } doneLabel: // Build the unions if required if (prcBounds && prcIcon && prcLabel) { ListView_CalcBounds(plv, QUERY_UNFOLDED, prcIcon, prcLabel, prcBounds); } if (prcSelectBounds && prcIcon && prcLabel) { if (ListView_HideLabels(plv)) *prcBounds = *prcIcon; else UnionRect(prcSelectBounds, prcIcon, prcLabel); } return fRc; } void ListView_InvalidateMark(LV* plv) { RECT rc; if (ListView_OnGetInsertMarkRect(plv, &rc)) { OffsetRect(&rc, -plv->ptOrigin.x, -plv->ptOrigin.y); InvalidateRect(plv->ci.hwnd, &rc, TRUE); } } // Returns the insertmark rect in listview coordinates. Returns FALSE if there is no insertmarkrect BOOL ListView_OnGetInsertMarkRect(LV* plv, LPRECT prc) { BOOL fVert; RECT rcSlot; LISTITEM *pitem; if (plv->iInsertItem == -1) return FALSE; pitem = ListView_GetItemPtr(plv, plv->iInsertItem); if (!pitem) { return FALSE; } fVert = !((plv->ci.style & LVS_ALIGNMASK) == LVS_ALIGNTOP); ListView_CalcItemSlotAndRect(plv, pitem, NULL, &rcSlot); if (fVert) { int iY; prc->left = rcSlot.left; prc->right = rcSlot.right; iY = (plv->fInsertAfter) ? rcSlot.bottom : rcSlot.top; prc->top = iY - INSERTMARKSIZE/2; prc->bottom = iY + INSERTMARKSIZE/2 + 1; } else { int iX; prc->top = rcSlot.top; prc->bottom = rcSlot.bottom; iX = (plv->fInsertAfter) ? rcSlot.right : rcSlot.left; prc->left = iX - INSERTMARKSIZE/2; prc->right = iX + INSERTMARKSIZE/2 + 1; } return TRUE; } COLORREF ListView_OnGetInsertMarkColor(LV* plv) { if (plv->clrim == CLR_DEFAULT) return plv->clrText; else return plv->clrim; }