|
|
@@ -1,1315 +1,1316 @@
|
|
|
-// QListCtrl.cpp : implementation file
|
|
|
-//
|
|
|
-
|
|
|
-#include "stdafx.h"
|
|
|
-#include "CP_Main.h"
|
|
|
-#include "QListCtrl.h"
|
|
|
-#include "ProcessPaste.h"
|
|
|
-#include "BitmapHelper.h"
|
|
|
-#include "MainTableFunctions.h"
|
|
|
-#include "DittoCopyBuffer.h"
|
|
|
-#include <atlbase.h>
|
|
|
-
|
|
|
-#ifdef _DEBUG
|
|
|
-#define new DEBUG_NEW
|
|
|
-#undef THIS_FILE
|
|
|
-static char THIS_FILE[] = __FILE__;
|
|
|
-#endif
|
|
|
-
|
|
|
-#define ROW_BOTTOM_BORDER 2
|
|
|
-#define ROW_LEFT_BORDER 3
|
|
|
-#define COLOR_SHADOW RGB(245, 245, 245)
|
|
|
-#define DUMMY_COL_WIDTH 1
|
|
|
-
|
|
|
-/////////////////////////////////////////////////////////////////////////////
|
|
|
-// CQListCtrl
|
|
|
-
|
|
|
-CQListCtrl::CQListCtrl()
|
|
|
-{
|
|
|
- m_pchTip = NULL;
|
|
|
- m_pwchTip = NULL;
|
|
|
- m_linesPerRow = 0;
|
|
|
-
|
|
|
- LOGFONT lf;
|
|
|
-
|
|
|
- lf.lfHeight = -9;
|
|
|
- lf.lfWidth = 0;
|
|
|
- lf.lfEscapement = 0;
|
|
|
- lf.lfOrientation = 0;
|
|
|
- lf.lfWeight = FW_LIGHT;
|
|
|
- lf.lfItalic = FALSE;
|
|
|
- lf.lfUnderline = FALSE;
|
|
|
- lf.lfStrikeOut = FALSE;
|
|
|
- lf.lfCharSet = ANSI_CHARSET;
|
|
|
- lf.lfOutPrecision = OUT_STRING_PRECIS;
|
|
|
- lf.lfClipPrecision = CLIP_STROKE_PRECIS;
|
|
|
- lf.lfQuality = DEFAULT_QUALITY;
|
|
|
- lf.lfPitchAndFamily = VARIABLE_PITCH | FF_DONTCARE;
|
|
|
- lstrcpy(lf.lfFaceName, _T("Small Font"));
|
|
|
-
|
|
|
- m_SmallFont = ::CreateFontIndirect(&lf);
|
|
|
-
|
|
|
- m_bShowTextForFirstTenHotKeys = true;
|
|
|
- m_bStartTop = true;
|
|
|
- m_pToolTip = NULL;
|
|
|
- m_pFormatter = NULL;
|
|
|
- m_allSelected = false;
|
|
|
-}
|
|
|
-
|
|
|
-CQListCtrl::~CQListCtrl()
|
|
|
-{
|
|
|
- if(m_pchTip != NULL)
|
|
|
- delete m_pchTip;
|
|
|
-
|
|
|
- if(m_pwchTip != NULL)
|
|
|
- delete m_pwchTip;
|
|
|
-
|
|
|
- if( m_SmallFont )
|
|
|
- ::DeleteObject( m_SmallFont );
|
|
|
-
|
|
|
- m_Font.DeleteObject();
|
|
|
-
|
|
|
- if(m_pFormatter)
|
|
|
- {
|
|
|
- delete m_pFormatter;
|
|
|
- m_pFormatter = NULL;
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-// returns the position 1-10 if the index is in the FirstTen block else -1
|
|
|
-int CQListCtrl::GetFirstTenNum( int index )
|
|
|
-{
|
|
|
- // set firstTenNum to the first ten number (1-10) corresponding to the given index
|
|
|
- int firstTenNum = -1; // -1 means that nItem is not in the FirstTen block.
|
|
|
- int count = GetItemCount();
|
|
|
-
|
|
|
- if( m_bStartTop )
|
|
|
- {
|
|
|
- if( 0 <= index && index <= 9 )
|
|
|
- firstTenNum = index + 1;
|
|
|
- }
|
|
|
- else // we are starting at the bottom and going up
|
|
|
- {
|
|
|
- int idxStartFirstTen = count-10; // start of the FirstTen block
|
|
|
- // if index is within the FirstTen block
|
|
|
- if( idxStartFirstTen <= index && index < count )
|
|
|
- firstTenNum = count - index;
|
|
|
- }
|
|
|
- return firstTenNum;
|
|
|
-}
|
|
|
-
|
|
|
-// returns the list index corresponding to the given FirstTen position number.
|
|
|
-// (ret < 0) means that "num" is not in the FirstTen block
|
|
|
-int CQListCtrl::GetFirstTenIndex( int num )
|
|
|
-{
|
|
|
- if( num <= 0 || num > 10 )
|
|
|
- return -1;
|
|
|
-
|
|
|
- if( m_bStartTop )
|
|
|
- return num-1;
|
|
|
- // else we are starting at the bottom and going up
|
|
|
- int count = GetItemCount();
|
|
|
- return count - num;
|
|
|
-}
|
|
|
-
|
|
|
-
|
|
|
-BEGIN_MESSAGE_MAP(CQListCtrl, CListCtrl)
|
|
|
-//{{AFX_MSG_MAP(CQListCtrl)
|
|
|
- ON_NOTIFY_REFLECT(LVN_KEYDOWN, OnKeydown)
|
|
|
- ON_NOTIFY_REFLECT(NM_DBLCLK, OnDblclk)
|
|
|
- ON_NOTIFY_REFLECT(NM_CUSTOMDRAW, OnCustomdrawList)
|
|
|
- ON_WM_SYSKEYDOWN()
|
|
|
- ON_WM_ERASEBKGND()
|
|
|
- ON_WM_CREATE()
|
|
|
- ON_WM_HSCROLL()
|
|
|
- ON_WM_TIMER()
|
|
|
- ON_NOTIFY_REFLECT(LVN_ITEMCHANGED, OnSelectionChange)
|
|
|
- ON_WM_VSCROLL()
|
|
|
- ON_WM_WINDOWPOSCHANGED()
|
|
|
- ON_WM_MOUSEWHEEL()
|
|
|
- //}}AFX_MSG_MAP
|
|
|
- ON_NOTIFY_EX_RANGE(TTN_NEEDTEXTW, 0, 0xFFFF, OnToolTipText)
|
|
|
- ON_NOTIFY_EX_RANGE(TTN_NEEDTEXTA, 0, 0xFFFF, OnToolTipText)
|
|
|
-ON_WM_KILLFOCUS()
|
|
|
-END_MESSAGE_MAP()
|
|
|
-
|
|
|
-/////////////////////////////////////////////////////////////////////////////
|
|
|
-// CQListCtrl message handlers
|
|
|
-
|
|
|
-void CQListCtrl::OnKeydown(NMHDR* pNMHDR, LRESULT* pResult)
|
|
|
-{
|
|
|
- LV_KEYDOWN* pLVKeyDown = (LV_KEYDOWN*)pNMHDR;
|
|
|
-
|
|
|
- switch (pLVKeyDown->wVKey)
|
|
|
- {
|
|
|
- case VK_RETURN:
|
|
|
- {
|
|
|
- ARRAY arr;
|
|
|
- GetSelectionIndexes(arr);
|
|
|
- SendSelection(arr);
|
|
|
- }
|
|
|
- break;
|
|
|
-
|
|
|
- case VK_ESCAPE:
|
|
|
- GetParent()->SendMessage(NM_END, 0, 0);
|
|
|
- break;
|
|
|
-
|
|
|
- case VK_RIGHT:
|
|
|
- {
|
|
|
- int nItem = GetNextItem(-1, LVNI_SELECTED);
|
|
|
- if (nItem != -1)
|
|
|
- GetParent()->SendMessage(NM_RIGHT, nItem, 0);
|
|
|
- }
|
|
|
- break;
|
|
|
-
|
|
|
- case VK_LEFT:
|
|
|
- GetParent()->SendMessage(NM_LEFT, 0, 0);
|
|
|
- break;
|
|
|
- case VK_DELETE:
|
|
|
- GetParent()->SendMessage(NM_DELETE, 0, 0);
|
|
|
- break;
|
|
|
- }
|
|
|
-
|
|
|
- *pResult = 0;
|
|
|
-}
|
|
|
-
|
|
|
-void CQListCtrl::OnDblclk(NMHDR* pNMHDR, LRESULT* pResult)
|
|
|
-{
|
|
|
- LPNMITEMACTIVATE lpnmItem = (LPNMITEMACTIVATE) pNMHDR;
|
|
|
-
|
|
|
- UINT Flags;
|
|
|
- int nItem = -1;
|
|
|
- if ((nItem = HitTest(lpnmItem->ptAction, &Flags)) != -1)
|
|
|
- {
|
|
|
- if (Flags | LVHT_ONITEM)
|
|
|
- SendSelection(nItem);
|
|
|
- }
|
|
|
-
|
|
|
- *pResult = 0;
|
|
|
-}
|
|
|
-
|
|
|
-void CQListCtrl::SendSelection(int nItem)
|
|
|
-{
|
|
|
- GetParent()->SendMessage(NM_SELECT, 1, (LPARAM) &nItem);
|
|
|
-}
|
|
|
-
|
|
|
-void CQListCtrl::SendSelection(ARRAY &arrItems)
|
|
|
-{
|
|
|
- GetParent()->SendMessage(NM_SELECT, arrItems.GetSize(), (LPARAM) arrItems.GetData());
|
|
|
-}
|
|
|
-
|
|
|
-void CQListCtrl::GetSelectionIndexes(ARRAY &arr)
|
|
|
-{
|
|
|
- arr.RemoveAll();
|
|
|
-
|
|
|
- POSITION pos = GetFirstSelectedItemPosition();
|
|
|
- while (pos)
|
|
|
- {
|
|
|
- arr.Add(GetNextSelectedItem(pos));
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-bool CQListCtrl::PutSelectedItemOnDittoCopyBuffer(long lBuffer)
|
|
|
-{
|
|
|
- bool bRet = false;
|
|
|
- ARRAY arr;
|
|
|
- GetSelectionItemData(arr);
|
|
|
- INT_PTR nCount = arr.GetSize();
|
|
|
- if(nCount > 0 && arr[0])
|
|
|
- {
|
|
|
- CDittoCopyBuffer::PutClipOnDittoCopyBuffer(arr[0], lBuffer);
|
|
|
- bRet = true;
|
|
|
- }
|
|
|
-
|
|
|
- return bRet;
|
|
|
-}
|
|
|
-
|
|
|
-void CQListCtrl::GetSelectionItemData(ARRAY &arr)
|
|
|
-{
|
|
|
- DWORD dwData;
|
|
|
- int i;
|
|
|
- arr.RemoveAll();
|
|
|
- POSITION pos = GetFirstSelectedItemPosition();
|
|
|
- while (pos)
|
|
|
- {
|
|
|
- i = GetNextSelectedItem(pos);
|
|
|
- dwData = GetItemData(i);
|
|
|
- arr.Add( dwData );
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-void CQListCtrl::RemoveAllSelection()
|
|
|
-{
|
|
|
- POSITION pos = GetFirstSelectedItemPosition();
|
|
|
- while (pos)
|
|
|
- {
|
|
|
- SetSelection(GetNextSelectedItem(pos), FALSE);
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-BOOL CQListCtrl::SetSelection(int nRow, BOOL bSelect)
|
|
|
-{
|
|
|
- if(bSelect)
|
|
|
- return SetItemState(nRow, LVIS_SELECTED, LVIS_SELECTED);
|
|
|
- else
|
|
|
- return SetItemState(nRow, ~LVIS_SELECTED, LVIS_SELECTED);
|
|
|
-}
|
|
|
-
|
|
|
-BOOL CQListCtrl::SetText(int nRow, int nCol, CString cs)
|
|
|
-{
|
|
|
- return SetItemText(nRow, nCol, cs);
|
|
|
-}
|
|
|
-
|
|
|
-BOOL CQListCtrl::SetCaret(int nRow, BOOL bFocus)
|
|
|
-{
|
|
|
- if(bFocus)
|
|
|
- return SetItemState(nRow, LVIS_FOCUSED, LVIS_FOCUSED);
|
|
|
- else
|
|
|
- return SetItemState(nRow, ~LVIS_FOCUSED, LVIS_FOCUSED);
|
|
|
-}
|
|
|
-
|
|
|
-long CQListCtrl::GetCaret()
|
|
|
-{
|
|
|
- return GetNextItem(-1, LVNI_FOCUSED);
|
|
|
-}
|
|
|
-
|
|
|
-// moves the caret to the given index, selects it, and ensures it is visible.
|
|
|
-BOOL CQListCtrl::SetListPos( int index )
|
|
|
-{
|
|
|
- if( index < 0 || index >= GetItemCount() )
|
|
|
- return FALSE;
|
|
|
-
|
|
|
- RemoveAllSelection();
|
|
|
- SetCaret(index);
|
|
|
- SetSelection(index);
|
|
|
- EnsureVisible(index,FALSE);
|
|
|
-
|
|
|
- return TRUE;
|
|
|
-}
|
|
|
-
|
|
|
-BOOL CQListCtrl::SetFormattedText(int nRow, int nCol, LPCTSTR lpszFormat,...)
|
|
|
-{
|
|
|
- CString csText;
|
|
|
- va_list vlist;
|
|
|
-
|
|
|
- ASSERT(AfxIsValidString(lpszFormat));
|
|
|
- va_start(vlist,lpszFormat);
|
|
|
- csText.FormatV(lpszFormat,vlist);
|
|
|
- va_end(vlist);
|
|
|
-
|
|
|
- return SetText(nRow,nCol,csText);
|
|
|
-}
|
|
|
-
|
|
|
-void CQListCtrl::SetNumberOfLinesPerRow(int nLines)
|
|
|
-{
|
|
|
- if(m_linesPerRow != nLines)
|
|
|
- {
|
|
|
- m_linesPerRow = nLines;
|
|
|
-
|
|
|
- CDC *pDC = GetDC();
|
|
|
-
|
|
|
- CRect crRect(0, 0, 0, 0);
|
|
|
-
|
|
|
- CFont *pOldFont = pDC->SelectObject(GetFont());
|
|
|
-
|
|
|
- //Get the height to draw one character
|
|
|
- pDC->DrawText("W", crRect, DT_VCENTER | DT_EXPANDTABS | DT_CALCRECT);
|
|
|
-
|
|
|
- pDC->SelectObject(pOldFont);
|
|
|
-
|
|
|
- //Get the total height of each row
|
|
|
- int nHeight = (crRect.Height() * nLines) + ROW_BOTTOM_BORDER;
|
|
|
-
|
|
|
- //Create a image list of that height and set it to the list box
|
|
|
- CImageList imglist;
|
|
|
- imglist.Create(DUMMY_COL_WIDTH, nHeight, ILC_COLOR16 | ILC_MASK, 1, 1);
|
|
|
- SetImageList(&imglist, LVSIL_SMALL );
|
|
|
-
|
|
|
- ReleaseDC(pDC);
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-void CQListCtrl::OnCustomdrawList(NMHDR* pNMHDR, LRESULT* pResult)
|
|
|
-{
|
|
|
- NMLVCUSTOMDRAW* pLVCD = reinterpret_cast<NMLVCUSTOMDRAW*>( pNMHDR );
|
|
|
-
|
|
|
- *pResult = 0;
|
|
|
-
|
|
|
- // Request item-specific notifications if this is the
|
|
|
- // beginning of the paint cycle.
|
|
|
- if ( CDDS_PREPAINT == pLVCD->nmcd.dwDrawStage )
|
|
|
- {
|
|
|
- *pResult = CDRF_NOTIFYITEMDRAW;
|
|
|
- }
|
|
|
- else if ( CDDS_ITEMPREPAINT == pLVCD->nmcd.dwDrawStage )
|
|
|
- {
|
|
|
- LVITEM rItem;
|
|
|
- int nItem = static_cast<int>( pLVCD->nmcd.dwItemSpec );
|
|
|
- CDC* pDC = CDC::FromHandle ( pLVCD->nmcd.hdc );
|
|
|
- COLORREF crBkgnd;
|
|
|
- BOOL bListHasFocus;
|
|
|
- CRect rcItem;
|
|
|
-
|
|
|
- bListHasFocus = ( GetSafeHwnd() == ::GetFocus() );
|
|
|
-
|
|
|
- // Get the image index and selected/focused state of the
|
|
|
- // item being drawn.
|
|
|
- ZeroMemory ( &rItem, sizeof(LVITEM) );
|
|
|
- rItem.mask = LVIF_STATE;
|
|
|
- rItem.iItem = nItem;
|
|
|
- rItem.stateMask = LVIS_SELECTED | LVIS_FOCUSED;
|
|
|
- GetItem(&rItem);
|
|
|
-
|
|
|
- // Get the rect that bounds the text label.
|
|
|
- GetItemRect(nItem, rcItem, LVIR_LABEL);
|
|
|
- rcItem.left -= DUMMY_COL_WIDTH;
|
|
|
-
|
|
|
- COLORREF OldColor = -1;
|
|
|
- int nOldBKMode = -1;
|
|
|
-
|
|
|
- // Draw the background of the list item. Colors are selected
|
|
|
- // according to the item's state.
|
|
|
- if(rItem.state & LVIS_SELECTED)
|
|
|
- {
|
|
|
- if(bListHasFocus)
|
|
|
- {
|
|
|
- crBkgnd = g_Opt.m_Theme.ListBoxSelectedBG();
|
|
|
- OldColor = pDC->SetTextColor(g_Opt.m_Theme.ListBoxSelectedText());
|
|
|
- }
|
|
|
- else
|
|
|
- {
|
|
|
- crBkgnd = g_Opt.m_Theme.ListBoxSelectedNoFocusBG();
|
|
|
- OldColor = pDC->SetTextColor(g_Opt.m_Theme.ListBoxSelectedNoFocusText());
|
|
|
- }
|
|
|
- }
|
|
|
- else
|
|
|
- {
|
|
|
- //Shade alternating Rows
|
|
|
- if((nItem % 2) == 0)
|
|
|
- {
|
|
|
- crBkgnd = g_Opt.m_Theme.ListBoxOddRowsBG();
|
|
|
- OldColor = pDC->SetTextColor(g_Opt.m_Theme.ListBoxOddRowsText());
|
|
|
- }
|
|
|
- else
|
|
|
- {
|
|
|
- crBkgnd = g_Opt.m_Theme.ListBoxEvenRowsBG();
|
|
|
- OldColor = pDC->SetTextColor(g_Opt.m_Theme.ListBoxEvenRowsText());
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- pDC->FillSolidRect(rcItem, crBkgnd);
|
|
|
- nOldBKMode = pDC->SetBkMode(TRANSPARENT);
|
|
|
-
|
|
|
- CRect rcText = rcItem;
|
|
|
- rcText.left += ROW_LEFT_BORDER;
|
|
|
- rcText.top++;
|
|
|
-
|
|
|
- // Draw the text.
|
|
|
- //CString csText = GetItemText(nItem, 0);
|
|
|
-
|
|
|
- CString csText;
|
|
|
- LPTSTR lpszText = csText.GetBufferSetLength(g_Opt.m_bDescTextSize);
|
|
|
- GetItemText(nItem, 0, lpszText, g_Opt.m_bDescTextSize);
|
|
|
- csText.ReleaseBuffer();
|
|
|
-
|
|
|
- // extract symbols
|
|
|
- CString strSymbols;
|
|
|
- int nSymEnd = csText.Find('|');
|
|
|
- if( nSymEnd >= 0 )
|
|
|
- {
|
|
|
- strSymbols = csText.Left(nSymEnd);
|
|
|
- csText = csText.Mid(nSymEnd+1);
|
|
|
- }
|
|
|
-
|
|
|
- // set firstTenNum to the first ten number (1-10) corresponding to
|
|
|
- // the current nItem.
|
|
|
- // -1 means that nItem is not in the FirstTen block.
|
|
|
- int firstTenNum = GetFirstTenNum(nItem);
|
|
|
-
|
|
|
- if( m_bShowTextForFirstTenHotKeys && firstTenNum > 0 )
|
|
|
- {
|
|
|
- rcText.left += 12;
|
|
|
- }
|
|
|
-
|
|
|
- // if we are inside a group, don't display the "in group" flag
|
|
|
- if( theApp.m_GroupID > 0 )
|
|
|
- {
|
|
|
- int nFlag = strSymbols.Find(_T("!"));
|
|
|
- if( nFlag >= 0 )
|
|
|
- strSymbols.Delete(nFlag);
|
|
|
- }
|
|
|
-
|
|
|
- DrawBitMap(nItem, rcText, pDC, csText);
|
|
|
-
|
|
|
- // draw the symbol box
|
|
|
- if( strSymbols.GetLength() > 0 )
|
|
|
- {
|
|
|
- strSymbols = " " + strSymbols + " "; // leave space for box
|
|
|
- // add spaces to leave room for the symbols
|
|
|
- CRect rectSym(rcText.left, rcText.top+1, rcText.left, rcText.top+1);
|
|
|
- CRect rectSpace(0,0,0,0);
|
|
|
- //Get text bounds
|
|
|
- pDC->DrawText(" ", &rectSpace, DT_VCENTER | DT_EXPANDTABS | DT_CALCRECT);
|
|
|
- pDC->DrawText(strSymbols, &rectSym, DT_VCENTER | DT_EXPANDTABS | DT_CALCRECT);
|
|
|
- VERIFY( rectSpace.Width() > 0 );
|
|
|
-
|
|
|
-// int numSpaces = rectSym.Width() / rectSpace.Width();
|
|
|
-// numSpaces++;
|
|
|
-// csText = CString(' ',numSpaces) + csText;
|
|
|
-
|
|
|
- // draw the symbols
|
|
|
- pDC->FillSolidRect( rectSym, GetSysColor(COLOR_ACTIVECAPTION) );
|
|
|
- //pDC->FillSolidRect( rectSym, RGB(0,255,255) );
|
|
|
- pDC->Draw3dRect(rectSym, GetSysColor(COLOR_3DLIGHT), GetSysColor(COLOR_3DDKSHADOW));
|
|
|
- // COLORREF crOld = pDC->SetTextColor(GetSysColor(COLOR_INFOTEXT));
|
|
|
- COLORREF crOld = pDC->SetTextColor(RGB(255, 255, 255));
|
|
|
- pDC->DrawText(strSymbols, rectSym, DT_VCENTER|DT_EXPANDTABS|DT_NOPREFIX);
|
|
|
- pDC->SetTextColor(crOld);
|
|
|
-
|
|
|
- rcText.left += rectSym.Width() + 2;
|
|
|
- }
|
|
|
-
|
|
|
- if(DrawRtfText(nItem, rcText, pDC) == FALSE)
|
|
|
- {
|
|
|
- pDC->DrawText(csText, rcText, DT_VCENTER|DT_EXPANDTABS|DT_NOPREFIX);
|
|
|
- }
|
|
|
-
|
|
|
- // Draw a focus rect around the item if necessary.
|
|
|
- if(bListHasFocus && (rItem.state & LVIS_FOCUSED))
|
|
|
- pDC->DrawFocusRect(rcItem);
|
|
|
-
|
|
|
- if( m_bShowTextForFirstTenHotKeys && firstTenNum > 0 )
|
|
|
- {
|
|
|
- CString cs;
|
|
|
- if( firstTenNum == 10 )
|
|
|
- cs = "0";
|
|
|
- else
|
|
|
- cs.Format(_T("%d"), firstTenNum);
|
|
|
-
|
|
|
- CRect crClient;
|
|
|
-
|
|
|
- GetWindowRect(crClient);
|
|
|
- ScreenToClient(crClient);
|
|
|
-
|
|
|
- CRect crHotKey = rcItem;
|
|
|
-
|
|
|
- crHotKey.right = crHotKey.left + 11;
|
|
|
- crHotKey.left += 2;
|
|
|
- crHotKey.top += 2;
|
|
|
-
|
|
|
- HFONT hOldFont = (HFONT)pDC->SelectObject(m_SmallFont);
|
|
|
-
|
|
|
- pDC->DrawText(cs, crHotKey, DT_BOTTOM);
|
|
|
-
|
|
|
- pDC->MoveTo(CPoint(rcItem.left + 11, rcItem.top));
|
|
|
- pDC->LineTo(CPoint(rcItem.left + 11, rcItem.bottom));
|
|
|
-
|
|
|
- pDC->SelectObject(hOldFont);
|
|
|
- }
|
|
|
-
|
|
|
- // restore the previous values
|
|
|
- if(OldColor > -1)
|
|
|
- pDC->SetTextColor(OldColor);
|
|
|
-
|
|
|
- if(nOldBKMode > -1)
|
|
|
- pDC->SetBkMode(nOldBKMode);
|
|
|
-
|
|
|
- *pResult = CDRF_SKIPDEFAULT; // We've painted everything.
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-BOOL CQListCtrl::DrawRtfText(int nItem, CRect &crRect, CDC *pDC)
|
|
|
-{
|
|
|
- if(g_Opt.m_bDrawRTF == FALSE)
|
|
|
- return FALSE;
|
|
|
-
|
|
|
- BOOL bRet = FALSE;
|
|
|
-
|
|
|
- CClipFormat* pThumbnail = GetItem_CF_RTF_ClipFormat(nItem);
|
|
|
- if(pThumbnail == NULL)
|
|
|
- return FALSE;
|
|
|
-
|
|
|
- // if there's no data, then we're done.
|
|
|
- if(pThumbnail->m_hgData == NULL)
|
|
|
- return FALSE;
|
|
|
-
|
|
|
- if(m_pFormatter == NULL)
|
|
|
- {
|
|
|
- m_pFormatter = new CFormattedTextDraw;
|
|
|
- m_pFormatter->Create();
|
|
|
- }
|
|
|
-
|
|
|
- if(m_pFormatter)
|
|
|
- {
|
|
|
- char *pData = (char*)GlobalLock(pThumbnail->m_hgData);
|
|
|
- if(pData)
|
|
|
- {
|
|
|
- CComBSTR bStr(pData);
|
|
|
-
|
|
|
- m_pFormatter->put_RTFText(bStr);
|
|
|
-
|
|
|
- m_pFormatter->Draw(pDC->m_hDC, crRect);
|
|
|
-
|
|
|
- GlobalUnlock(pThumbnail->m_hgData);
|
|
|
-
|
|
|
- bRet = TRUE;
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- return bRet;
|
|
|
-}
|
|
|
-
|
|
|
-// DrawBitMap loads a DIB from the DB, draws a crRect thumbnail of the image
|
|
|
-// to pDC and caches that thumbnail as a DIB in m_ThumbNails[ ItemID ].
|
|
|
-// ALL items are cached in m_ThumbNails (those without images are cached with NULL m_hgData)
|
|
|
-BOOL CQListCtrl::DrawBitMap(int nItem, CRect &crRect, CDC *pDC, const CString &csDescription)
|
|
|
-{
|
|
|
- if(g_Opt.m_bDrawThumbnail == FALSE)
|
|
|
- return FALSE;
|
|
|
-
|
|
|
- CClipFormatQListCtrl *format = GetItem_CF_DIB_ClipFormat(nItem);
|
|
|
- if(format != NULL)
|
|
|
- {
|
|
|
- HGLOBAL smallImage = format->GetDib(pDC, crRect.Height());
|
|
|
- if(smallImage != NULL)
|
|
|
- {
|
|
|
- //Will return the width of the bitmap in nWidth
|
|
|
- int nWidth = 0;
|
|
|
- if(CBitmapHelper::DrawDIB(pDC, smallImage, crRect.left, crRect.top, nWidth))
|
|
|
- {
|
|
|
- // adjust the rect so other information can be drawn next to the thumbnail
|
|
|
- crRect.left += nWidth + 3;
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
- else if(csDescription.Find(_T("CF_DIB")) == 0)
|
|
|
- {
|
|
|
- crRect.left += crRect.Height();
|
|
|
- }
|
|
|
-
|
|
|
- return TRUE;
|
|
|
-}
|
|
|
-
|
|
|
-void CQListCtrl::RefreshVisibleRows()
|
|
|
-{
|
|
|
- int nTopIndex = GetTopIndex();
|
|
|
- int nLastIndex = nTopIndex + GetCountPerPage();
|
|
|
- RedrawItems(nTopIndex, nLastIndex);
|
|
|
-}
|
|
|
-
|
|
|
-void CQListCtrl::RefreshRow(int row)
|
|
|
-{
|
|
|
- RedrawItems(row, row);
|
|
|
-}
|
|
|
-
|
|
|
-void CQListCtrl::OnSysKeyDown(UINT nChar, UINT nRepCnt, UINT nFlags)
|
|
|
-{
|
|
|
- if(GetKeyState(VK_RETURN) & 0x800)
|
|
|
- GetParent()->SendMessage(NM_PROPERTIES, 0, 0);
|
|
|
- else
|
|
|
- CListCtrl::OnSysKeyDown(nChar, nRepCnt, nFlags);
|
|
|
-}
|
|
|
-
|
|
|
-BOOL CQListCtrl::OnEraseBkgnd(CDC* pDC)
|
|
|
-{
|
|
|
-
|
|
|
- // Simply returning TRUE seems OK since we do custom item
|
|
|
- // painting. However, there is a pixel buffer around the
|
|
|
- // border of this control (not within the item rects)
|
|
|
- // which becomes visually corrupt if it is not erased.
|
|
|
-
|
|
|
- // In most cases, I do not notice the erasure, so I have kept
|
|
|
- // the call to CListCtrl::OnEraseBkgnd(pDC);
|
|
|
-
|
|
|
- // However, for some reason, bulk erasure is very noticeable when
|
|
|
- // shift-scrolling the page to select a block of items, so
|
|
|
- // I made a special case for that:
|
|
|
- if(GetSelectedCount() >= 2)
|
|
|
- return TRUE;
|
|
|
- return CListCtrl::OnEraseBkgnd(pDC);
|
|
|
-}
|
|
|
-
|
|
|
-BOOL CQListCtrl::OnToolTipText( UINT id, NMHDR * pNMHDR, LRESULT * pResult )
|
|
|
-{
|
|
|
- // need to handle both ANSI and UNICODE versions of the message
|
|
|
- TOOLTIPTEXTA* pTTTA = (TOOLTIPTEXTA*)pNMHDR;
|
|
|
- TOOLTIPTEXTW* pTTTW = (TOOLTIPTEXTW*)pNMHDR;
|
|
|
- CString strTipText;
|
|
|
-
|
|
|
- UINT_PTR nID = pNMHDR->idFrom;
|
|
|
-
|
|
|
- if(nID == 0) // Notification in NT from automatically
|
|
|
- return FALSE; // created tooltip
|
|
|
-
|
|
|
- ::SendMessage(pNMHDR->hwndFrom, TTM_SETMAXTIPWIDTH, 0, 500);
|
|
|
-
|
|
|
- // Use Item's name as the tool tip. Change this for something different.
|
|
|
- // Like use its file size, etc.
|
|
|
- GetToolTipText((int)nID-1, strTipText);
|
|
|
-
|
|
|
- //Replace the tabs with spaces, the tooltip didn't like the \t s
|
|
|
- strTipText.Replace(_T("\t"), _T(" "));
|
|
|
-
|
|
|
- int nLength = strTipText.GetLength()+2;
|
|
|
-
|
|
|
-#ifndef _UNICODE
|
|
|
- if (pNMHDR->code == TTN_NEEDTEXTA)
|
|
|
- {
|
|
|
- if(m_pchTip != NULL)
|
|
|
- delete m_pchTip;
|
|
|
-
|
|
|
- m_pchTip = new TCHAR[nLength];
|
|
|
- lstrcpyn(m_pchTip, strTipText, nLength-1);
|
|
|
- m_pchTip[nLength-1] = 0;
|
|
|
- pTTTW->lpszText = (WCHAR*)m_pchTip;
|
|
|
- }
|
|
|
- else
|
|
|
- {
|
|
|
- if(m_pwchTip != NULL)
|
|
|
- delete m_pwchTip;
|
|
|
-
|
|
|
- m_pwchTip = new WCHAR[nLength];
|
|
|
- _mbstowcsz(m_pwchTip, strTipText, nLength-1);
|
|
|
- m_pwchTip[nLength-1] = 0; // end of text
|
|
|
- pTTTW->lpszText = (WCHAR*)m_pwchTip;
|
|
|
- }
|
|
|
-#else
|
|
|
- if(pNMHDR->code == TTN_NEEDTEXTA)
|
|
|
- {
|
|
|
- if(m_pchTip != NULL)
|
|
|
- delete m_pchTip;
|
|
|
-
|
|
|
- m_pchTip = new TCHAR[nLength];
|
|
|
- STRNCPY(m_pchTip, strTipText, nLength-1);
|
|
|
- m_pchTip[nLength-1] = 0; // end of text
|
|
|
- pTTTW->lpszText = (LPTSTR)m_pchTip;
|
|
|
- }
|
|
|
- else
|
|
|
- {
|
|
|
- if(m_pwchTip != NULL)
|
|
|
- delete m_pwchTip;
|
|
|
-
|
|
|
- m_pwchTip = new WCHAR[nLength];
|
|
|
- lstrcpyn(m_pwchTip, strTipText, nLength-1);
|
|
|
- m_pwchTip[nLength-1] = 0;
|
|
|
- pTTTW->lpszText = (LPTSTR) m_pwchTip;
|
|
|
- }
|
|
|
-#endif
|
|
|
- *pResult = 0;
|
|
|
-
|
|
|
- return TRUE; // message was handled
|
|
|
-}
|
|
|
-
|
|
|
-INT_PTR CQListCtrl::OnToolHitTest(CPoint point, TOOLINFO * pTI) const
|
|
|
-{
|
|
|
- CRect rect;
|
|
|
- GetClientRect(&rect);
|
|
|
- if(rect.PtInRect(point))
|
|
|
- {
|
|
|
- if(GetItemCount())
|
|
|
- {
|
|
|
- int nTopIndex = GetTopIndex();
|
|
|
- int nBottomIndex = nTopIndex + GetCountPerPage();
|
|
|
- if(nBottomIndex > GetItemCount()) nBottomIndex = GetItemCount();
|
|
|
- for(int nIndex = nTopIndex; nIndex <= nBottomIndex; nIndex++)
|
|
|
- {
|
|
|
- GetItemRect(nIndex, rect, LVIR_BOUNDS);
|
|
|
- if(rect.PtInRect(point))
|
|
|
- {
|
|
|
- pTI->hwnd = m_hWnd;
|
|
|
- pTI->uId = (UINT)(nIndex+1);
|
|
|
- pTI->lpszText = LPSTR_TEXTCALLBACK;
|
|
|
- pTI->rect = rect;
|
|
|
- return pTI->uId;
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- return -1;
|
|
|
-}
|
|
|
-
|
|
|
-int CQListCtrl::OnCreate(LPCREATESTRUCT lpCreateStruct)
|
|
|
-{
|
|
|
- if (CListCtrl::OnCreate(lpCreateStruct) == -1)
|
|
|
- return -1;
|
|
|
-
|
|
|
- EnableToolTips();
|
|
|
-
|
|
|
- m_pToolTip = new CToolTipEx;
|
|
|
- m_pToolTip->Create(this);
|
|
|
-
|
|
|
- m_pToolTip->SetNotifyWnd(GetParent());
|
|
|
-
|
|
|
- return 0;
|
|
|
-}
|
|
|
-
|
|
|
-BOOL CQListCtrl::PreTranslateMessage(MSG* pMsg)
|
|
|
-{
|
|
|
- DWORD dID;
|
|
|
- if(m_Accels.OnMsg(pMsg, dID))
|
|
|
- {
|
|
|
- switch(dID)
|
|
|
- {
|
|
|
- case COPY_BUFFER_HOT_KEY_1_ID:
|
|
|
- PutSelectedItemOnDittoCopyBuffer(0);
|
|
|
- break;
|
|
|
- case COPY_BUFFER_HOT_KEY_2_ID:
|
|
|
- PutSelectedItemOnDittoCopyBuffer(1);
|
|
|
- break;
|
|
|
- case COPY_BUFFER_HOT_KEY_3_ID:
|
|
|
- PutSelectedItemOnDittoCopyBuffer(2);
|
|
|
- break;
|
|
|
- default:
|
|
|
- GetParent()->SendMessage(NM_SELECT_DB_ID, dID, 0);
|
|
|
- }
|
|
|
-
|
|
|
- return TRUE;
|
|
|
- }
|
|
|
-
|
|
|
- if(m_pToolTip)
|
|
|
- {
|
|
|
- if(m_pToolTip->OnMsg(pMsg))
|
|
|
- return TRUE;
|
|
|
- }
|
|
|
-
|
|
|
- switch(pMsg->message)
|
|
|
- {
|
|
|
- case WM_KEYDOWN:
|
|
|
- if(HandleKeyDown(pMsg->wParam, pMsg->lParam))
|
|
|
- return TRUE;
|
|
|
-
|
|
|
- break; // end case WM_KEYDOWN
|
|
|
-
|
|
|
- case WM_VSCROLL:
|
|
|
- ASSERT(FALSE);
|
|
|
- break;
|
|
|
- } // end switch(pMsg->message)
|
|
|
-
|
|
|
- return CListCtrl::PreTranslateMessage(pMsg);
|
|
|
-}
|
|
|
-
|
|
|
-BOOL CQListCtrl::HandleKeyDown(WPARAM wParam, LPARAM lParam)
|
|
|
-{
|
|
|
- if(m_pToolTip)
|
|
|
- {
|
|
|
- MSG Msg;
|
|
|
- Msg.lParam = lParam;
|
|
|
- Msg.wParam = wParam;
|
|
|
- Msg.message = WM_KEYDOWN;
|
|
|
- if(m_pToolTip->OnMsg(&Msg))
|
|
|
- return TRUE;
|
|
|
- }
|
|
|
-
|
|
|
- WPARAM vk = wParam;
|
|
|
-
|
|
|
- // if a number key was pressed
|
|
|
- if('0' <= vk && vk <= '9')
|
|
|
- {
|
|
|
- // if <Ctrl> is required but is absent, then break
|
|
|
- if(g_Opt.m_bUseCtrlNumAccel && !(GetKeyState(VK_CONTROL) & 0x8000))
|
|
|
- return FALSE;
|
|
|
-
|
|
|
- int index = (int)vk - '0';
|
|
|
- // '0' is actually 10 in the ditto window
|
|
|
- if(index == 0)
|
|
|
- index = 10;
|
|
|
- // translate num 1-10 into the actual index (based upon m_bStartTop)
|
|
|
- index = GetFirstTenIndex(index);
|
|
|
- GetParent()->SendMessage(NM_SELECT_INDEX, index, 0);
|
|
|
- return TRUE;
|
|
|
- }
|
|
|
-
|
|
|
- if(VK_NUMPAD0 <= vk && vk <= VK_NUMPAD9)
|
|
|
- {
|
|
|
- // if <Ctrl> is required but is absent, then break
|
|
|
- if( g_Opt.m_bUseCtrlNumAccel && !(GetKeyState(VK_CONTROL) & 0x8000) )
|
|
|
- return FALSE;
|
|
|
-
|
|
|
- int index = (int)vk - VK_NUMPAD0;
|
|
|
- // '0' is actually 10 in the ditto window
|
|
|
- if(index == 0)
|
|
|
- index = 10;
|
|
|
-
|
|
|
- // translate num 1-10 into the actual index (based upon m_bStartTop)
|
|
|
- index = GetFirstTenIndex(index);
|
|
|
- GetParent()->SendMessage(NM_SELECT_INDEX, index, 0);
|
|
|
- return TRUE;
|
|
|
- }
|
|
|
-
|
|
|
- switch( vk )
|
|
|
- {
|
|
|
- case 'X': // Ctrl-X = Cut (prepare for moving the items into a Group)
|
|
|
- if(GetKeyState(VK_CONTROL) & 0x8000)
|
|
|
- {
|
|
|
- LoadCopyOrCutToClipboard();
|
|
|
-
|
|
|
- theApp.IC_Cut(); // uses selection
|
|
|
- return TRUE;
|
|
|
- }
|
|
|
- break;
|
|
|
-
|
|
|
- case 'C': // Ctrl-C = Copy (prepare for copying the items into a Group)
|
|
|
- if(GetKeyState(VK_CONTROL) & 0x8000)
|
|
|
- {
|
|
|
- LoadCopyOrCutToClipboard();
|
|
|
-
|
|
|
- theApp.IC_Copy(); // uses selection
|
|
|
- return TRUE;
|
|
|
- }
|
|
|
- break;
|
|
|
-
|
|
|
- case 'V': // Ctrl-V = Paste (actually performs the copy or move of items into the current Group)
|
|
|
- if(GetKeyState(VK_CONTROL) & 0x8000)
|
|
|
- {
|
|
|
- theApp.IC_Paste();
|
|
|
- return TRUE;
|
|
|
- }
|
|
|
- break;
|
|
|
-
|
|
|
- case 'A': // Ctrl-A = Select All
|
|
|
- if(GetKeyState(VK_CONTROL) & 0x8000)
|
|
|
- {
|
|
|
- int nCount = GetItemCount();
|
|
|
- for(int i = 0; i < nCount; i++)
|
|
|
- {
|
|
|
- SetSelection(i);
|
|
|
- }
|
|
|
- return TRUE;
|
|
|
- }
|
|
|
- break;
|
|
|
-
|
|
|
- case VK_F3:
|
|
|
- {
|
|
|
- ShowFullDescription();
|
|
|
- return TRUE;
|
|
|
- }
|
|
|
- case VK_BACK:
|
|
|
- theApp.EnterGroupID( theApp.m_GroupParentID );
|
|
|
- return TRUE;
|
|
|
- case VK_SPACE:
|
|
|
- if(GetKeyState(VK_CONTROL) & 0x8000)
|
|
|
- {
|
|
|
- theApp.ShowPersistent( !g_Opt.m_bShowPersistent );
|
|
|
- return TRUE;
|
|
|
- }
|
|
|
- break;
|
|
|
- } // end switch(vk)
|
|
|
-
|
|
|
- return FALSE;
|
|
|
-}
|
|
|
-
|
|
|
-void CQListCtrl::LoadCopyOrCutToClipboard()
|
|
|
-{
|
|
|
- ARRAY arr;
|
|
|
- GetSelectionItemData(arr);
|
|
|
- INT_PTR count = arr.GetSize();
|
|
|
- if(count <= 0)
|
|
|
- return;
|
|
|
-
|
|
|
- CProcessPaste paste;
|
|
|
-
|
|
|
- //Don't send the paste just load it into memory
|
|
|
- paste.m_bSendPaste = false;
|
|
|
-
|
|
|
- if(count > 1)
|
|
|
- paste.GetClipIDs().Copy(arr);
|
|
|
- else
|
|
|
- paste.GetClipIDs().Add(arr[0]);
|
|
|
-
|
|
|
- //Don't move these to the top
|
|
|
- BOOL bItWas = g_Opt.m_bUpdateTimeOnPaste;
|
|
|
- g_Opt.m_bUpdateTimeOnPaste = FALSE;
|
|
|
-
|
|
|
- paste.DoPaste();
|
|
|
-
|
|
|
- g_Opt.m_bUpdateTimeOnPaste = bItWas;
|
|
|
-}
|
|
|
-
|
|
|
-void CQListCtrl::ShowFullDescription(bool bFromAuto)
|
|
|
-{
|
|
|
- int nItem = GetCaret();
|
|
|
- CRect rc, crWindow;
|
|
|
- GetWindowRect(&crWindow);
|
|
|
- GetItemRect(nItem, rc, LVIR_BOUNDS);
|
|
|
- ClientToScreen(rc);
|
|
|
-
|
|
|
- CPoint pt;
|
|
|
-
|
|
|
- if(bFromAuto == false)
|
|
|
- {
|
|
|
- pt = CPoint(rc.left, rc.bottom);
|
|
|
- }
|
|
|
- else
|
|
|
- pt = CPoint((crWindow.left + (crWindow.right - crWindow.left)/2), rc.bottom);
|
|
|
-
|
|
|
- CString csDescription;
|
|
|
- GetToolTipText(nItem, csDescription);
|
|
|
-
|
|
|
- m_pToolTip->DestroyWindow();
|
|
|
-
|
|
|
- m_pToolTip = new CToolTipEx;
|
|
|
- m_pToolTip->Create(this);
|
|
|
- m_pToolTip->SetNotifyWnd(GetParent());
|
|
|
-
|
|
|
- if(m_pToolTip)
|
|
|
- {
|
|
|
- m_pToolTip->SetToolTipText(_T(""));
|
|
|
- m_pToolTip->SetRTFText(" ");
|
|
|
- bool bSetPlainText = false;
|
|
|
- CClipFormat Clip;
|
|
|
-
|
|
|
- Clip.m_cfType = CF_UNICODETEXT;
|
|
|
- if(GetClipData(nItem, Clip) && Clip.m_hgData)
|
|
|
- {
|
|
|
- LPVOID pvData = GlobalLock(Clip.m_hgData);
|
|
|
- if(pvData)
|
|
|
- {
|
|
|
- CString csText = (WCHAR*)pvData;
|
|
|
- m_pToolTip->SetToolTipText(csText);
|
|
|
- bSetPlainText = true;
|
|
|
- }
|
|
|
-
|
|
|
- GlobalUnlock(Clip.m_hgData);
|
|
|
-
|
|
|
- Clip.Free();
|
|
|
- Clip.Clear();
|
|
|
- }
|
|
|
-
|
|
|
- if(bSetPlainText == false)
|
|
|
- {
|
|
|
- Clip.m_cfType = CF_TEXT;
|
|
|
- if(GetClipData(nItem, Clip) && Clip.m_hgData)
|
|
|
- {
|
|
|
- LPVOID pvData = GlobalLock(Clip.m_hgData);
|
|
|
- if(pvData)
|
|
|
- {
|
|
|
- CString csText = (char*)pvData;
|
|
|
- m_pToolTip->SetToolTipText(csText);
|
|
|
-
|
|
|
- bSetPlainText = true;
|
|
|
- }
|
|
|
-
|
|
|
- GlobalUnlock(Clip.m_hgData);
|
|
|
-
|
|
|
- Clip.Free();
|
|
|
- Clip.Clear();
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- if(bSetPlainText == false)
|
|
|
- {
|
|
|
- m_pToolTip->SetToolTipText(csDescription);
|
|
|
- }
|
|
|
-
|
|
|
- Clip.m_cfType = RegisterClipboardFormat(CF_RTF);
|
|
|
-
|
|
|
- if(GetClipData(nItem, Clip) && Clip.m_hgData)
|
|
|
- {
|
|
|
- LPVOID pvData = GlobalLock(Clip.m_hgData);
|
|
|
- if(pvData)
|
|
|
- {
|
|
|
- m_pToolTip->SetRTFText((char*)pvData);
|
|
|
- }
|
|
|
-
|
|
|
- GlobalUnlock(Clip.m_hgData);
|
|
|
-
|
|
|
- Clip.Free();
|
|
|
- Clip.Clear();
|
|
|
- }
|
|
|
-
|
|
|
- Clip.m_cfType = CF_DIB;
|
|
|
-
|
|
|
- if(GetClipData(nItem, Clip) && Clip.m_hgData)
|
|
|
- {
|
|
|
- CBitmap *pBitMap = new CBitmap;
|
|
|
- if(pBitMap)
|
|
|
- {
|
|
|
- CRect rcItem;
|
|
|
- GetWindowRect(rcItem);
|
|
|
-
|
|
|
- CDC *pDC = GetDC();;
|
|
|
-
|
|
|
- CBitmapHelper::GetCBitmap(&Clip, pDC, pBitMap, (rcItem.Width() * 2));
|
|
|
-
|
|
|
- ReleaseDC(pDC);
|
|
|
-
|
|
|
- //Tooltip wnd will release
|
|
|
- m_pToolTip->SetBitmap(pBitMap);
|
|
|
- }
|
|
|
-
|
|
|
- Clip.Free();
|
|
|
- Clip.Clear();
|
|
|
- }
|
|
|
-
|
|
|
-
|
|
|
- m_pToolTip->Show(pt);
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-void CQListCtrl::GetToolTipText(int nItem, CString &csText)
|
|
|
-{
|
|
|
- CWnd* pParent=GetParent();
|
|
|
- if(pParent && (pParent->GetSafeHwnd() != NULL))
|
|
|
- {
|
|
|
- CQListToolTipText info;
|
|
|
- memset(&info, 0, sizeof(info));
|
|
|
- info.hdr.code = NM_GETTOOLTIPTEXT;
|
|
|
- info.hdr.hwndFrom = GetSafeHwnd();
|
|
|
- info.hdr.idFrom = GetDlgCtrlID();
|
|
|
- info.lItem = nItem;
|
|
|
- //plus 100 for extra info - shortcut and such
|
|
|
- info.cchTextMax = g_Opt.m_bDescTextSize + 100;
|
|
|
- info.pszText = csText.GetBufferSetLength(info.cchTextMax);
|
|
|
-
|
|
|
- pParent->SendMessage(WM_NOTIFY,(WPARAM)info.hdr.idFrom,(LPARAM)&info);
|
|
|
-
|
|
|
- csText.ReleaseBuffer();
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-BOOL CQListCtrl::GetClipData(int nItem, CClipFormat &Clip)
|
|
|
-{
|
|
|
- return theApp.GetClipData(GetItemData(nItem), Clip);
|
|
|
-}
|
|
|
-
|
|
|
-DWORD CQListCtrl::GetItemData(int nItem)
|
|
|
-{
|
|
|
- if((GetStyle() & LVS_OWNERDATA))
|
|
|
- {
|
|
|
- CWnd* pParent=GetParent();
|
|
|
- if(pParent && (pParent->GetSafeHwnd() != NULL))
|
|
|
- {
|
|
|
- LV_DISPINFO info;
|
|
|
- memset(&info, 0, sizeof(info));
|
|
|
- info.hdr.code = LVN_GETDISPINFO;
|
|
|
- info.hdr.hwndFrom = GetSafeHwnd();
|
|
|
- info.hdr.idFrom = GetDlgCtrlID();
|
|
|
-
|
|
|
- info.item.iItem = nItem;
|
|
|
- info.item.lParam = -1;
|
|
|
- info.item.mask = LVIF_PARAM;
|
|
|
-
|
|
|
- pParent->SendMessage(WM_NOTIFY,(WPARAM)info.hdr.idFrom,(LPARAM)&info);
|
|
|
-
|
|
|
- return (DWORD)info.item.lParam;
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- return (DWORD)CListCtrl::GetItemData(nItem);
|
|
|
-}
|
|
|
-
|
|
|
-CClipFormatQListCtrl* CQListCtrl::GetItem_CF_DIB_ClipFormat(int nItem)
|
|
|
-{
|
|
|
- CClipFormatQListCtrl *format = NULL;
|
|
|
-
|
|
|
- CWnd* pParent=GetParent();
|
|
|
- if(pParent && (pParent->GetSafeHwnd() != NULL))
|
|
|
- {
|
|
|
- LV_DISPINFO info;
|
|
|
- memset(&info, 0, sizeof(info));
|
|
|
- info.hdr.code = LVN_GETDISPINFO;
|
|
|
- info.hdr.hwndFrom = GetSafeHwnd();
|
|
|
- info.hdr.idFrom = GetDlgCtrlID();
|
|
|
-
|
|
|
- info.item.iItem = nItem;
|
|
|
- info.item.lParam = NULL;
|
|
|
- info.item.mask = LVIF_CF_DIB;
|
|
|
-
|
|
|
- pParent->SendMessage(WM_NOTIFY,(WPARAM)info.hdr.idFrom,(LPARAM)&info);
|
|
|
-
|
|
|
- if(info.item.lParam != NULL)
|
|
|
- {
|
|
|
- format = (CClipFormatQListCtrl *)info.item.lParam;
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- return format;
|
|
|
-}
|
|
|
-
|
|
|
-CClipFormatQListCtrl* CQListCtrl::GetItem_CF_RTF_ClipFormat(int nItem)
|
|
|
-{
|
|
|
- CClipFormatQListCtrl *format = NULL;
|
|
|
-
|
|
|
- CWnd* pParent=GetParent();
|
|
|
- if(pParent && (pParent->GetSafeHwnd() != NULL))
|
|
|
- {
|
|
|
- LV_DISPINFO info;
|
|
|
- memset(&info, 0, sizeof(info));
|
|
|
- info.hdr.code = LVN_GETDISPINFO;
|
|
|
- info.hdr.hwndFrom = GetSafeHwnd();
|
|
|
- info.hdr.idFrom = GetDlgCtrlID();
|
|
|
-
|
|
|
- info.item.iItem = nItem;
|
|
|
- info.item.lParam = NULL;
|
|
|
- info.item.mask = LVIF_CF_RICHTEXT;
|
|
|
-
|
|
|
- pParent->SendMessage(WM_NOTIFY, (WPARAM)info.hdr.idFrom, (LPARAM)&info);
|
|
|
-
|
|
|
- if(info.item.lParam != NULL)
|
|
|
- {
|
|
|
- format = (CClipFormatQListCtrl *)info.item.lParam;
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- return format;
|
|
|
-}
|
|
|
-
|
|
|
-void CQListCtrl::OnHScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar)
|
|
|
-{
|
|
|
- CListCtrl::OnHScroll(nSBCode, nPos, pScrollBar);
|
|
|
-}
|
|
|
-
|
|
|
-void CQListCtrl::DestroyAndCreateAccelerator(BOOL bCreate, CppSQLite3DB &db)
|
|
|
-{
|
|
|
- m_Accels.m_Map.RemoveAll();
|
|
|
-
|
|
|
- if(bCreate)
|
|
|
- {
|
|
|
- CMainTableFunctions::LoadAcceleratorKeys(m_Accels, db);
|
|
|
-
|
|
|
- LoadDittoCopyBufferHotkeys();
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-void CQListCtrl::LoadDittoCopyBufferHotkeys()
|
|
|
-{
|
|
|
- CCopyBufferItem Item;
|
|
|
- CAccel a;
|
|
|
-
|
|
|
- g_Opt.GetCopyBufferItem(0, Item);
|
|
|
- if(Item.m_lCopyHotKey > 0)
|
|
|
- {
|
|
|
- a.Cmd = COPY_BUFFER_HOT_KEY_1_ID;
|
|
|
- a.Key = Item.m_lCopyHotKey;
|
|
|
- m_Accels.AddAccel(a);
|
|
|
- }
|
|
|
-
|
|
|
- g_Opt.GetCopyBufferItem(1, Item);
|
|
|
- if(Item.m_lCopyHotKey > 0)
|
|
|
- {
|
|
|
- a.Cmd = COPY_BUFFER_HOT_KEY_2_ID;
|
|
|
- a.Key = Item.m_lCopyHotKey;
|
|
|
- m_Accels.AddAccel(a);
|
|
|
- }
|
|
|
-
|
|
|
- g_Opt.GetCopyBufferItem(2, Item);
|
|
|
- if(Item.m_lCopyHotKey > 0)
|
|
|
- {
|
|
|
- a.Cmd = COPY_BUFFER_HOT_KEY_3_ID;
|
|
|
- a.Key = Item.m_lCopyHotKey;
|
|
|
- m_Accels.AddAccel(a);
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-void CQListCtrl::OnKillFocus(CWnd* pNewWnd)
|
|
|
-{
|
|
|
- CListCtrl::OnKillFocus(pNewWnd);
|
|
|
-
|
|
|
-// if(FocusOnToolTip() == FALSE)
|
|
|
-// m_pToolTip->Hide();
|
|
|
-}
|
|
|
-
|
|
|
-HWND CQListCtrl::GetToolTipHWnd()
|
|
|
-{
|
|
|
- return m_pToolTip->GetSafeHwnd();
|
|
|
-}
|
|
|
-
|
|
|
-BOOL CQListCtrl::SetItemCountEx(int iCount, DWORD dwFlags /* = 0 */)
|
|
|
-{
|
|
|
- return CListCtrl::SetItemCountEx(iCount, dwFlags);
|
|
|
-}
|
|
|
-
|
|
|
-#define TIMER_SHOW_PROPERTIES 1
|
|
|
-
|
|
|
-void CQListCtrl::OnSelectionChange(NMHDR* pNMHDR, LRESULT* pResult)
|
|
|
-{
|
|
|
- NMLISTVIEW *pnmv = (NMLISTVIEW *) pNMHDR;
|
|
|
-
|
|
|
- if((pnmv->uNewState == 3) ||
|
|
|
- (pnmv->uNewState == 1))
|
|
|
- {
|
|
|
- if(g_Opt.m_bAllwaysShowDescription)
|
|
|
- {
|
|
|
- KillTimer(TIMER_SHOW_PROPERTIES);
|
|
|
- SetTimer(TIMER_SHOW_PROPERTIES, 300, NULL);
|
|
|
- }
|
|
|
- if(GetSelectedCount() > 0 )
|
|
|
- theApp.SetStatus(NULL, FALSE);
|
|
|
- }
|
|
|
-
|
|
|
- if(GetSelectedCount() == this->GetItemCount())
|
|
|
- {
|
|
|
- if(m_allSelected == false)
|
|
|
- {
|
|
|
- Log(StrF(_T("List box Select All")));
|
|
|
-
|
|
|
- GetParent()->SendMessage(NM_ALL_SELECTED, 0, 0);
|
|
|
- m_allSelected = true;
|
|
|
- }
|
|
|
- }
|
|
|
- else if(m_allSelected == true)
|
|
|
- {
|
|
|
- Log(StrF(_T("List box REMOVED Select All")));
|
|
|
- m_allSelected = false;
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-void CQListCtrl::OnTimer(UINT_PTR nIDEvent)
|
|
|
-{
|
|
|
- if(nIDEvent == TIMER_SHOW_PROPERTIES)
|
|
|
- {
|
|
|
- if( theApp.m_bShowingQuickPaste )
|
|
|
- ShowFullDescription(true);
|
|
|
- KillTimer(TIMER_SHOW_PROPERTIES);
|
|
|
- }
|
|
|
-
|
|
|
- CListCtrl::OnTimer(nIDEvent);
|
|
|
-}
|
|
|
-
|
|
|
-void CQListCtrl::SetLogFont(LOGFONT &font)
|
|
|
-{
|
|
|
- m_Font.DeleteObject();
|
|
|
-
|
|
|
- m_Font.CreateFontIndirect(&font);
|
|
|
-
|
|
|
- SetFont(&m_Font);
|
|
|
-}
|
|
|
-
|
|
|
-void CQListCtrl::OnVScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar)
|
|
|
-{
|
|
|
- CListCtrl::OnVScroll(nSBCode, nPos, pScrollBar);
|
|
|
-}
|
|
|
-
|
|
|
-BOOL CQListCtrl::OnMouseWheel(UINT nFlags, short zDelta, CPoint pt)
|
|
|
-{
|
|
|
- return CListCtrl::OnMouseWheel(nFlags, zDelta, pt);
|
|
|
-}
|
|
|
-
|
|
|
-BOOL CQListCtrl::OnChildNotify(UINT message, WPARAM wParam, LPARAM lParam, LRESULT* pLResult)
|
|
|
-{
|
|
|
- NMLVCACHEHINT* pcachehint = NULL;
|
|
|
-
|
|
|
- if(message == WM_NOTIFY)
|
|
|
- {
|
|
|
- NMHDR* phdr = (NMHDR*)lParam;
|
|
|
-
|
|
|
- switch(phdr->code)
|
|
|
- {
|
|
|
- case LVN_ODCACHEHINT:
|
|
|
- pcachehint= (NMLVCACHEHINT*) phdr;
|
|
|
-
|
|
|
- GetParent()->SendMessage(NM_FILL_REST_OF_LIST, pcachehint->iFrom, pcachehint->iTo);
|
|
|
- return FALSE;
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- return CListCtrl::OnChildNotify(message, wParam, lParam, pLResult);
|
|
|
-}
|
|
|
-
|
|
|
-BOOL CQListCtrl::OnItemDeleted(long lID)
|
|
|
-{
|
|
|
- BOOL bRet2 = m_RTFData.RemoveKey(lID);
|
|
|
-
|
|
|
- return (bRet2);
|
|
|
+// QListCtrl.cpp : implementation file
|
|
|
+//
|
|
|
+
|
|
|
+#include "stdafx.h"
|
|
|
+#include "CP_Main.h"
|
|
|
+#include "QListCtrl.h"
|
|
|
+#include "ProcessPaste.h"
|
|
|
+#include "BitmapHelper.h"
|
|
|
+#include "MainTableFunctions.h"
|
|
|
+#include "DittoCopyBuffer.h"
|
|
|
+#include <atlbase.h>
|
|
|
+
|
|
|
+#ifdef _DEBUG
|
|
|
+#define new DEBUG_NEW
|
|
|
+#undef THIS_FILE
|
|
|
+static char THIS_FILE[] = __FILE__;
|
|
|
+#endif
|
|
|
+
|
|
|
+#define ROW_BOTTOM_BORDER 2
|
|
|
+#define ROW_LEFT_BORDER 3
|
|
|
+#define COLOR_SHADOW RGB(245, 245, 245)
|
|
|
+#define DUMMY_COL_WIDTH 1
|
|
|
+
|
|
|
+/////////////////////////////////////////////////////////////////////////////
|
|
|
+// CQListCtrl
|
|
|
+
|
|
|
+CQListCtrl::CQListCtrl()
|
|
|
+{
|
|
|
+ m_pchTip = NULL;
|
|
|
+ m_pwchTip = NULL;
|
|
|
+ m_linesPerRow = 1;
|
|
|
+
|
|
|
+ LOGFONT lf;
|
|
|
+
|
|
|
+ lf.lfHeight = -9;
|
|
|
+ lf.lfWidth = 0;
|
|
|
+ lf.lfEscapement = 0;
|
|
|
+ lf.lfOrientation = 0;
|
|
|
+ lf.lfWeight = FW_LIGHT;
|
|
|
+ lf.lfItalic = FALSE;
|
|
|
+ lf.lfUnderline = FALSE;
|
|
|
+ lf.lfStrikeOut = FALSE;
|
|
|
+ lf.lfCharSet = ANSI_CHARSET;
|
|
|
+ lf.lfOutPrecision = OUT_STRING_PRECIS;
|
|
|
+ lf.lfClipPrecision = CLIP_STROKE_PRECIS;
|
|
|
+ lf.lfQuality = DEFAULT_QUALITY;
|
|
|
+ lf.lfPitchAndFamily = VARIABLE_PITCH | FF_DONTCARE;
|
|
|
+ lstrcpy(lf.lfFaceName, _T("Small Font"));
|
|
|
+
|
|
|
+ m_SmallFont = ::CreateFontIndirect(&lf);
|
|
|
+
|
|
|
+ m_bShowTextForFirstTenHotKeys = true;
|
|
|
+ m_bStartTop = true;
|
|
|
+ m_pToolTip = NULL;
|
|
|
+ m_pFormatter = NULL;
|
|
|
+ m_allSelected = false;
|
|
|
+}
|
|
|
+
|
|
|
+CQListCtrl::~CQListCtrl()
|
|
|
+{
|
|
|
+ if(m_pchTip != NULL)
|
|
|
+ delete m_pchTip;
|
|
|
+
|
|
|
+ if(m_pwchTip != NULL)
|
|
|
+ delete m_pwchTip;
|
|
|
+
|
|
|
+ if( m_SmallFont )
|
|
|
+ ::DeleteObject( m_SmallFont );
|
|
|
+
|
|
|
+ m_Font.DeleteObject();
|
|
|
+
|
|
|
+ if(m_pFormatter)
|
|
|
+ {
|
|
|
+ delete m_pFormatter;
|
|
|
+ m_pFormatter = NULL;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+// returns the position 1-10 if the index is in the FirstTen block else -1
|
|
|
+int CQListCtrl::GetFirstTenNum( int index )
|
|
|
+{
|
|
|
+ // set firstTenNum to the first ten number (1-10) corresponding to the given index
|
|
|
+ int firstTenNum = -1; // -1 means that nItem is not in the FirstTen block.
|
|
|
+ int count = GetItemCount();
|
|
|
+
|
|
|
+ if( m_bStartTop )
|
|
|
+ {
|
|
|
+ if( 0 <= index && index <= 9 )
|
|
|
+ firstTenNum = index + 1;
|
|
|
+ }
|
|
|
+ else // we are starting at the bottom and going up
|
|
|
+ {
|
|
|
+ int idxStartFirstTen = count-10; // start of the FirstTen block
|
|
|
+ // if index is within the FirstTen block
|
|
|
+ if( idxStartFirstTen <= index && index < count )
|
|
|
+ firstTenNum = count - index;
|
|
|
+ }
|
|
|
+ return firstTenNum;
|
|
|
+}
|
|
|
+
|
|
|
+// returns the list index corresponding to the given FirstTen position number.
|
|
|
+// (ret < 0) means that "num" is not in the FirstTen block
|
|
|
+int CQListCtrl::GetFirstTenIndex( int num )
|
|
|
+{
|
|
|
+ if( num <= 0 || num > 10 )
|
|
|
+ return -1;
|
|
|
+
|
|
|
+ if( m_bStartTop )
|
|
|
+ return num-1;
|
|
|
+ // else we are starting at the bottom and going up
|
|
|
+ int count = GetItemCount();
|
|
|
+ return count - num;
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
+BEGIN_MESSAGE_MAP(CQListCtrl, CListCtrl)
|
|
|
+//{{AFX_MSG_MAP(CQListCtrl)
|
|
|
+ ON_NOTIFY_REFLECT(LVN_KEYDOWN, OnKeydown)
|
|
|
+ ON_NOTIFY_REFLECT(NM_DBLCLK, OnDblclk)
|
|
|
+ ON_NOTIFY_REFLECT(NM_CUSTOMDRAW, OnCustomdrawList)
|
|
|
+ ON_WM_SYSKEYDOWN()
|
|
|
+ ON_WM_ERASEBKGND()
|
|
|
+ ON_WM_CREATE()
|
|
|
+ ON_WM_HSCROLL()
|
|
|
+ ON_WM_TIMER()
|
|
|
+ ON_NOTIFY_REFLECT(LVN_ITEMCHANGED, OnSelectionChange)
|
|
|
+ ON_WM_VSCROLL()
|
|
|
+ ON_WM_WINDOWPOSCHANGED()
|
|
|
+ ON_WM_MOUSEWHEEL()
|
|
|
+ //}}AFX_MSG_MAP
|
|
|
+ ON_NOTIFY_EX_RANGE(TTN_NEEDTEXTW, 0, 0xFFFF, OnToolTipText)
|
|
|
+ ON_NOTIFY_EX_RANGE(TTN_NEEDTEXTA, 0, 0xFFFF, OnToolTipText)
|
|
|
+ ON_WM_KILLFOCUS()
|
|
|
+ ON_WM_MEASUREITEM_REFLECT()
|
|
|
+END_MESSAGE_MAP()
|
|
|
+
|
|
|
+/////////////////////////////////////////////////////////////////////////////
|
|
|
+// CQListCtrl message handlers
|
|
|
+
|
|
|
+void CQListCtrl::OnKeydown(NMHDR* pNMHDR, LRESULT* pResult)
|
|
|
+{
|
|
|
+ LV_KEYDOWN* pLVKeyDown = (LV_KEYDOWN*)pNMHDR;
|
|
|
+
|
|
|
+ switch (pLVKeyDown->wVKey)
|
|
|
+ {
|
|
|
+ case VK_RETURN:
|
|
|
+ {
|
|
|
+ ARRAY arr;
|
|
|
+ GetSelectionIndexes(arr);
|
|
|
+ SendSelection(arr);
|
|
|
+ }
|
|
|
+ break;
|
|
|
+
|
|
|
+ case VK_ESCAPE:
|
|
|
+ GetParent()->SendMessage(NM_END, 0, 0);
|
|
|
+ break;
|
|
|
+
|
|
|
+ case VK_RIGHT:
|
|
|
+ {
|
|
|
+ int nItem = GetNextItem(-1, LVNI_SELECTED);
|
|
|
+ if (nItem != -1)
|
|
|
+ GetParent()->SendMessage(NM_RIGHT, nItem, 0);
|
|
|
+ }
|
|
|
+ break;
|
|
|
+
|
|
|
+ case VK_LEFT:
|
|
|
+ GetParent()->SendMessage(NM_LEFT, 0, 0);
|
|
|
+ break;
|
|
|
+ case VK_DELETE:
|
|
|
+ GetParent()->SendMessage(NM_DELETE, 0, 0);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+
|
|
|
+ *pResult = 0;
|
|
|
+}
|
|
|
+
|
|
|
+void CQListCtrl::OnDblclk(NMHDR* pNMHDR, LRESULT* pResult)
|
|
|
+{
|
|
|
+ LPNMITEMACTIVATE lpnmItem = (LPNMITEMACTIVATE) pNMHDR;
|
|
|
+
|
|
|
+ UINT Flags;
|
|
|
+ int nItem = -1;
|
|
|
+ if ((nItem = HitTest(lpnmItem->ptAction, &Flags)) != -1)
|
|
|
+ {
|
|
|
+ if (Flags | LVHT_ONITEM)
|
|
|
+ SendSelection(nItem);
|
|
|
+ }
|
|
|
+
|
|
|
+ *pResult = 0;
|
|
|
+}
|
|
|
+
|
|
|
+void CQListCtrl::SendSelection(int nItem)
|
|
|
+{
|
|
|
+ GetParent()->SendMessage(NM_SELECT, 1, (LPARAM) &nItem);
|
|
|
+}
|
|
|
+
|
|
|
+void CQListCtrl::SendSelection(ARRAY &arrItems)
|
|
|
+{
|
|
|
+ GetParent()->SendMessage(NM_SELECT, arrItems.GetSize(), (LPARAM) arrItems.GetData());
|
|
|
+}
|
|
|
+
|
|
|
+void CQListCtrl::GetSelectionIndexes(ARRAY &arr)
|
|
|
+{
|
|
|
+ arr.RemoveAll();
|
|
|
+
|
|
|
+ POSITION pos = GetFirstSelectedItemPosition();
|
|
|
+ while (pos)
|
|
|
+ {
|
|
|
+ arr.Add(GetNextSelectedItem(pos));
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+bool CQListCtrl::PutSelectedItemOnDittoCopyBuffer(long lBuffer)
|
|
|
+{
|
|
|
+ bool bRet = false;
|
|
|
+ ARRAY arr;
|
|
|
+ GetSelectionItemData(arr);
|
|
|
+ INT_PTR nCount = arr.GetSize();
|
|
|
+ if(nCount > 0 && arr[0])
|
|
|
+ {
|
|
|
+ CDittoCopyBuffer::PutClipOnDittoCopyBuffer(arr[0], lBuffer);
|
|
|
+ bRet = true;
|
|
|
+ }
|
|
|
+
|
|
|
+ return bRet;
|
|
|
+}
|
|
|
+
|
|
|
+void CQListCtrl::GetSelectionItemData(ARRAY &arr)
|
|
|
+{
|
|
|
+ DWORD dwData;
|
|
|
+ int i;
|
|
|
+ arr.RemoveAll();
|
|
|
+ POSITION pos = GetFirstSelectedItemPosition();
|
|
|
+ while (pos)
|
|
|
+ {
|
|
|
+ i = GetNextSelectedItem(pos);
|
|
|
+ dwData = GetItemData(i);
|
|
|
+ arr.Add( dwData );
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+void CQListCtrl::RemoveAllSelection()
|
|
|
+{
|
|
|
+ POSITION pos = GetFirstSelectedItemPosition();
|
|
|
+ while (pos)
|
|
|
+ {
|
|
|
+ SetSelection(GetNextSelectedItem(pos), FALSE);
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+BOOL CQListCtrl::SetSelection(int nRow, BOOL bSelect)
|
|
|
+{
|
|
|
+ if(bSelect)
|
|
|
+ return SetItemState(nRow, LVIS_SELECTED, LVIS_SELECTED);
|
|
|
+ else
|
|
|
+ return SetItemState(nRow, ~LVIS_SELECTED, LVIS_SELECTED);
|
|
|
+}
|
|
|
+
|
|
|
+BOOL CQListCtrl::SetText(int nRow, int nCol, CString cs)
|
|
|
+{
|
|
|
+ return SetItemText(nRow, nCol, cs);
|
|
|
+}
|
|
|
+
|
|
|
+BOOL CQListCtrl::SetCaret(int nRow, BOOL bFocus)
|
|
|
+{
|
|
|
+ if(bFocus)
|
|
|
+ return SetItemState(nRow, LVIS_FOCUSED, LVIS_FOCUSED);
|
|
|
+ else
|
|
|
+ return SetItemState(nRow, ~LVIS_FOCUSED, LVIS_FOCUSED);
|
|
|
+}
|
|
|
+
|
|
|
+long CQListCtrl::GetCaret()
|
|
|
+{
|
|
|
+ return GetNextItem(-1, LVNI_FOCUSED);
|
|
|
+}
|
|
|
+
|
|
|
+// moves the caret to the given index, selects it, and ensures it is visible.
|
|
|
+BOOL CQListCtrl::SetListPos( int index )
|
|
|
+{
|
|
|
+ if( index < 0 || index >= GetItemCount() )
|
|
|
+ return FALSE;
|
|
|
+
|
|
|
+ RemoveAllSelection();
|
|
|
+ SetCaret(index);
|
|
|
+ SetSelection(index);
|
|
|
+ EnsureVisible(index,FALSE);
|
|
|
+
|
|
|
+ return TRUE;
|
|
|
+}
|
|
|
+
|
|
|
+BOOL CQListCtrl::SetFormattedText(int nRow, int nCol, LPCTSTR lpszFormat,...)
|
|
|
+{
|
|
|
+ CString csText;
|
|
|
+ va_list vlist;
|
|
|
+
|
|
|
+ ASSERT(AfxIsValidString(lpszFormat));
|
|
|
+ va_start(vlist,lpszFormat);
|
|
|
+ csText.FormatV(lpszFormat,vlist);
|
|
|
+ va_end(vlist);
|
|
|
+
|
|
|
+ return SetText(nRow,nCol,csText);
|
|
|
+}
|
|
|
+
|
|
|
+void CQListCtrl::SetNumberOfLinesPerRow(int nLines)
|
|
|
+{
|
|
|
+ if(m_linesPerRow != nLines)
|
|
|
+ {
|
|
|
+ m_linesPerRow = nLines;
|
|
|
+
|
|
|
+ CRect rc;
|
|
|
+ GetWindowRect( &rc );
|
|
|
+ WINDOWPOS wp;
|
|
|
+ wp.hwnd = m_hWnd;
|
|
|
+ wp.cx = rc.Width();
|
|
|
+ wp.cy = rc.Height();
|
|
|
+ wp.flags = SWP_NOACTIVATE | SWP_NOMOVE | SWP_NOOWNERZORDER | SWP_NOZORDER;
|
|
|
+ SendMessage( WM_WINDOWPOSCHANGED, 0, (LPARAM)&wp );
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+void CQListCtrl::MeasureItem(LPMEASUREITEMSTRUCT lpMeasureItemStruct)
|
|
|
+{
|
|
|
+ TEXTMETRIC tm;
|
|
|
+ HDC hDC = ::GetDC(NULL);
|
|
|
+ CFont* pFont = GetFont();
|
|
|
+ HFONT hFontOld = (HFONT)SelectObject(hDC, pFont->GetSafeHandle());
|
|
|
+ GetTextMetrics(hDC, &tm);
|
|
|
+ lpMeasureItemStruct->itemHeight = ((tm.tmHeight + tm.tmExternalLeading) * m_linesPerRow) + ROW_BOTTOM_BORDER;
|
|
|
+ SelectObject(hDC, hFontOld);
|
|
|
+ ::ReleaseDC(NULL, hDC);
|
|
|
+}
|
|
|
+
|
|
|
+void CQListCtrl::OnCustomdrawList(NMHDR* pNMHDR, LRESULT* pResult)
|
|
|
+{
|
|
|
+ NMLVCUSTOMDRAW* pLVCD = reinterpret_cast<NMLVCUSTOMDRAW*>( pNMHDR );
|
|
|
+
|
|
|
+ *pResult = 0;
|
|
|
+
|
|
|
+ // Request item-specific notifications if this is the
|
|
|
+ // beginning of the paint cycle.
|
|
|
+ if ( CDDS_PREPAINT == pLVCD->nmcd.dwDrawStage )
|
|
|
+ {
|
|
|
+ *pResult = CDRF_NOTIFYITEMDRAW;
|
|
|
+ }
|
|
|
+ else if ( CDDS_ITEMPREPAINT == pLVCD->nmcd.dwDrawStage )
|
|
|
+ {
|
|
|
+ LVITEM rItem;
|
|
|
+ int nItem = static_cast<int>( pLVCD->nmcd.dwItemSpec );
|
|
|
+ CDC* pDC = CDC::FromHandle ( pLVCD->nmcd.hdc );
|
|
|
+ COLORREF crBkgnd;
|
|
|
+ BOOL bListHasFocus;
|
|
|
+ CRect rcItem;
|
|
|
+
|
|
|
+ bListHasFocus = ( GetSafeHwnd() == ::GetFocus() );
|
|
|
+
|
|
|
+ // Get the image index and selected/focused state of the
|
|
|
+ // item being drawn.
|
|
|
+ ZeroMemory ( &rItem, sizeof(LVITEM) );
|
|
|
+ rItem.mask = LVIF_STATE;
|
|
|
+ rItem.iItem = nItem;
|
|
|
+ rItem.stateMask = LVIS_SELECTED | LVIS_FOCUSED;
|
|
|
+ GetItem(&rItem);
|
|
|
+
|
|
|
+ // Get the rect that bounds the text label.
|
|
|
+ GetItemRect(nItem, rcItem, LVIR_LABEL);
|
|
|
+ rcItem.left -= DUMMY_COL_WIDTH;
|
|
|
+
|
|
|
+ COLORREF OldColor = -1;
|
|
|
+ int nOldBKMode = -1;
|
|
|
+
|
|
|
+ // Draw the background of the list item. Colors are selected
|
|
|
+ // according to the item's state.
|
|
|
+ if(rItem.state & LVIS_SELECTED)
|
|
|
+ {
|
|
|
+ if(bListHasFocus)
|
|
|
+ {
|
|
|
+ crBkgnd = g_Opt.m_Theme.ListBoxSelectedBG();
|
|
|
+ OldColor = pDC->SetTextColor(g_Opt.m_Theme.ListBoxSelectedText());
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ crBkgnd = g_Opt.m_Theme.ListBoxSelectedNoFocusBG();
|
|
|
+ OldColor = pDC->SetTextColor(g_Opt.m_Theme.ListBoxSelectedNoFocusText());
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ //Shade alternating Rows
|
|
|
+ if((nItem % 2) == 0)
|
|
|
+ {
|
|
|
+ crBkgnd = g_Opt.m_Theme.ListBoxOddRowsBG();
|
|
|
+ OldColor = pDC->SetTextColor(g_Opt.m_Theme.ListBoxOddRowsText());
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ crBkgnd = g_Opt.m_Theme.ListBoxEvenRowsBG();
|
|
|
+ OldColor = pDC->SetTextColor(g_Opt.m_Theme.ListBoxEvenRowsText());
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ pDC->FillSolidRect(rcItem, crBkgnd);
|
|
|
+ nOldBKMode = pDC->SetBkMode(TRANSPARENT);
|
|
|
+
|
|
|
+ CRect rcText = rcItem;
|
|
|
+ rcText.left += ROW_LEFT_BORDER;
|
|
|
+ rcText.top++;
|
|
|
+
|
|
|
+ // Draw the text.
|
|
|
+ //CString csText = GetItemText(nItem, 0);
|
|
|
+
|
|
|
+ CString csText;
|
|
|
+ LPTSTR lpszText = csText.GetBufferSetLength(g_Opt.m_bDescTextSize);
|
|
|
+ GetItemText(nItem, 0, lpszText, g_Opt.m_bDescTextSize);
|
|
|
+ csText.ReleaseBuffer();
|
|
|
+
|
|
|
+ // extract symbols
|
|
|
+ CString strSymbols;
|
|
|
+ int nSymEnd = csText.Find('|');
|
|
|
+ if( nSymEnd >= 0 )
|
|
|
+ {
|
|
|
+ strSymbols = csText.Left(nSymEnd);
|
|
|
+ csText = csText.Mid(nSymEnd+1);
|
|
|
+ }
|
|
|
+
|
|
|
+ // set firstTenNum to the first ten number (1-10) corresponding to
|
|
|
+ // the current nItem.
|
|
|
+ // -1 means that nItem is not in the FirstTen block.
|
|
|
+ int firstTenNum = GetFirstTenNum(nItem);
|
|
|
+
|
|
|
+ if( m_bShowTextForFirstTenHotKeys && firstTenNum > 0 )
|
|
|
+ {
|
|
|
+ rcText.left += 12;
|
|
|
+ }
|
|
|
+
|
|
|
+ // if we are inside a group, don't display the "in group" flag
|
|
|
+ if( theApp.m_GroupID > 0 )
|
|
|
+ {
|
|
|
+ int nFlag = strSymbols.Find(_T("!"));
|
|
|
+ if( nFlag >= 0 )
|
|
|
+ strSymbols.Delete(nFlag);
|
|
|
+ }
|
|
|
+
|
|
|
+ DrawBitMap(nItem, rcText, pDC, csText);
|
|
|
+
|
|
|
+ // draw the symbol box
|
|
|
+ if( strSymbols.GetLength() > 0 )
|
|
|
+ {
|
|
|
+ strSymbols = " " + strSymbols + " "; // leave space for box
|
|
|
+ // add spaces to leave room for the symbols
|
|
|
+ CRect rectSym(rcText.left, rcText.top+1, rcText.left, rcText.top+1);
|
|
|
+ CRect rectSpace(0,0,0,0);
|
|
|
+ //Get text bounds
|
|
|
+ pDC->DrawText(" ", &rectSpace, DT_VCENTER | DT_EXPANDTABS | DT_CALCRECT);
|
|
|
+ pDC->DrawText(strSymbols, &rectSym, DT_VCENTER | DT_EXPANDTABS | DT_CALCRECT);
|
|
|
+ VERIFY( rectSpace.Width() > 0 );
|
|
|
+
|
|
|
+// int numSpaces = rectSym.Width() / rectSpace.Width();
|
|
|
+// numSpaces++;
|
|
|
+// csText = CString(' ',numSpaces) + csText;
|
|
|
+
|
|
|
+ // draw the symbols
|
|
|
+ pDC->FillSolidRect( rectSym, GetSysColor(COLOR_ACTIVECAPTION) );
|
|
|
+ //pDC->FillSolidRect( rectSym, RGB(0,255,255) );
|
|
|
+ pDC->Draw3dRect(rectSym, GetSysColor(COLOR_3DLIGHT), GetSysColor(COLOR_3DDKSHADOW));
|
|
|
+ // COLORREF crOld = pDC->SetTextColor(GetSysColor(COLOR_INFOTEXT));
|
|
|
+ COLORREF crOld = pDC->SetTextColor(RGB(255, 255, 255));
|
|
|
+ pDC->DrawText(strSymbols, rectSym, DT_VCENTER|DT_EXPANDTABS|DT_NOPREFIX);
|
|
|
+ pDC->SetTextColor(crOld);
|
|
|
+
|
|
|
+ rcText.left += rectSym.Width() + 2;
|
|
|
+ }
|
|
|
+
|
|
|
+ if(DrawRtfText(nItem, rcText, pDC) == FALSE)
|
|
|
+ {
|
|
|
+ pDC->DrawText(csText, rcText, DT_VCENTER|DT_EXPANDTABS|DT_NOPREFIX);
|
|
|
+ }
|
|
|
+
|
|
|
+ // Draw a focus rect around the item if necessary.
|
|
|
+ if(bListHasFocus && (rItem.state & LVIS_FOCUSED))
|
|
|
+ pDC->DrawFocusRect(rcItem);
|
|
|
+
|
|
|
+ if( m_bShowTextForFirstTenHotKeys && firstTenNum > 0 )
|
|
|
+ {
|
|
|
+ CString cs;
|
|
|
+ if( firstTenNum == 10 )
|
|
|
+ cs = "0";
|
|
|
+ else
|
|
|
+ cs.Format(_T("%d"), firstTenNum);
|
|
|
+
|
|
|
+ CRect crClient;
|
|
|
+
|
|
|
+ GetWindowRect(crClient);
|
|
|
+ ScreenToClient(crClient);
|
|
|
+
|
|
|
+ CRect crHotKey = rcItem;
|
|
|
+
|
|
|
+ crHotKey.right = crHotKey.left + 11;
|
|
|
+ crHotKey.left += 2;
|
|
|
+ crHotKey.top += 2;
|
|
|
+
|
|
|
+ HFONT hOldFont = (HFONT)pDC->SelectObject(m_SmallFont);
|
|
|
+
|
|
|
+ pDC->DrawText(cs, crHotKey, DT_BOTTOM);
|
|
|
+
|
|
|
+ pDC->MoveTo(CPoint(rcItem.left + 11, rcItem.top));
|
|
|
+ pDC->LineTo(CPoint(rcItem.left + 11, rcItem.bottom));
|
|
|
+
|
|
|
+ pDC->SelectObject(hOldFont);
|
|
|
+ }
|
|
|
+
|
|
|
+ // restore the previous values
|
|
|
+ if(OldColor > -1)
|
|
|
+ pDC->SetTextColor(OldColor);
|
|
|
+
|
|
|
+ if(nOldBKMode > -1)
|
|
|
+ pDC->SetBkMode(nOldBKMode);
|
|
|
+
|
|
|
+ *pResult = CDRF_SKIPDEFAULT; // We've painted everything.
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+BOOL CQListCtrl::DrawRtfText(int nItem, CRect &crRect, CDC *pDC)
|
|
|
+{
|
|
|
+ if(g_Opt.m_bDrawRTF == FALSE)
|
|
|
+ return FALSE;
|
|
|
+
|
|
|
+ BOOL bRet = FALSE;
|
|
|
+
|
|
|
+ CClipFormat* pThumbnail = GetItem_CF_RTF_ClipFormat(nItem);
|
|
|
+ if(pThumbnail == NULL)
|
|
|
+ return FALSE;
|
|
|
+
|
|
|
+ // if there's no data, then we're done.
|
|
|
+ if(pThumbnail->m_hgData == NULL)
|
|
|
+ return FALSE;
|
|
|
+
|
|
|
+ if(m_pFormatter == NULL)
|
|
|
+ {
|
|
|
+ m_pFormatter = new CFormattedTextDraw;
|
|
|
+ m_pFormatter->Create();
|
|
|
+ }
|
|
|
+
|
|
|
+ if(m_pFormatter)
|
|
|
+ {
|
|
|
+ char *pData = (char*)GlobalLock(pThumbnail->m_hgData);
|
|
|
+ if(pData)
|
|
|
+ {
|
|
|
+ CComBSTR bStr(pData);
|
|
|
+
|
|
|
+ m_pFormatter->put_RTFText(bStr);
|
|
|
+
|
|
|
+ m_pFormatter->Draw(pDC->m_hDC, crRect);
|
|
|
+
|
|
|
+ GlobalUnlock(pThumbnail->m_hgData);
|
|
|
+
|
|
|
+ bRet = TRUE;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ return bRet;
|
|
|
+}
|
|
|
+
|
|
|
+// DrawBitMap loads a DIB from the DB, draws a crRect thumbnail of the image
|
|
|
+// to pDC and caches that thumbnail as a DIB in m_ThumbNails[ ItemID ].
|
|
|
+// ALL items are cached in m_ThumbNails (those without images are cached with NULL m_hgData)
|
|
|
+BOOL CQListCtrl::DrawBitMap(int nItem, CRect &crRect, CDC *pDC, const CString &csDescription)
|
|
|
+{
|
|
|
+ if(g_Opt.m_bDrawThumbnail == FALSE)
|
|
|
+ return FALSE;
|
|
|
+
|
|
|
+ CClipFormatQListCtrl *format = GetItem_CF_DIB_ClipFormat(nItem);
|
|
|
+ if(format != NULL)
|
|
|
+ {
|
|
|
+ HGLOBAL smallImage = format->GetDib(pDC, crRect.Height());
|
|
|
+ if(smallImage != NULL)
|
|
|
+ {
|
|
|
+ //Will return the width of the bitmap in nWidth
|
|
|
+ int nWidth = 0;
|
|
|
+ if(CBitmapHelper::DrawDIB(pDC, smallImage, crRect.left, crRect.top, nWidth))
|
|
|
+ {
|
|
|
+ // adjust the rect so other information can be drawn next to the thumbnail
|
|
|
+ crRect.left += nWidth + 3;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else if(csDescription.Find(_T("CF_DIB")) == 0)
|
|
|
+ {
|
|
|
+ crRect.left += crRect.Height();
|
|
|
+ }
|
|
|
+
|
|
|
+ return TRUE;
|
|
|
+}
|
|
|
+
|
|
|
+void CQListCtrl::RefreshVisibleRows()
|
|
|
+{
|
|
|
+ int nTopIndex = GetTopIndex();
|
|
|
+ int nLastIndex = nTopIndex + GetCountPerPage();
|
|
|
+ RedrawItems(nTopIndex, nLastIndex);
|
|
|
+}
|
|
|
+
|
|
|
+void CQListCtrl::RefreshRow(int row)
|
|
|
+{
|
|
|
+ RedrawItems(row, row);
|
|
|
+}
|
|
|
+
|
|
|
+void CQListCtrl::OnSysKeyDown(UINT nChar, UINT nRepCnt, UINT nFlags)
|
|
|
+{
|
|
|
+ if(GetKeyState(VK_RETURN) & 0x800)
|
|
|
+ GetParent()->SendMessage(NM_PROPERTIES, 0, 0);
|
|
|
+ else
|
|
|
+ CListCtrl::OnSysKeyDown(nChar, nRepCnt, nFlags);
|
|
|
+}
|
|
|
+
|
|
|
+BOOL CQListCtrl::OnEraseBkgnd(CDC* pDC)
|
|
|
+{
|
|
|
+
|
|
|
+ // Simply returning TRUE seems OK since we do custom item
|
|
|
+ // painting. However, there is a pixel buffer around the
|
|
|
+ // border of this control (not within the item rects)
|
|
|
+ // which becomes visually corrupt if it is not erased.
|
|
|
+
|
|
|
+ // In most cases, I do not notice the erasure, so I have kept
|
|
|
+ // the call to CListCtrl::OnEraseBkgnd(pDC);
|
|
|
+
|
|
|
+ // However, for some reason, bulk erasure is very noticeable when
|
|
|
+ // shift-scrolling the page to select a block of items, so
|
|
|
+ // I made a special case for that:
|
|
|
+ if(GetSelectedCount() >= 2)
|
|
|
+ return TRUE;
|
|
|
+ return CListCtrl::OnEraseBkgnd(pDC);
|
|
|
+}
|
|
|
+
|
|
|
+BOOL CQListCtrl::OnToolTipText( UINT id, NMHDR * pNMHDR, LRESULT * pResult )
|
|
|
+{
|
|
|
+ // need to handle both ANSI and UNICODE versions of the message
|
|
|
+ TOOLTIPTEXTA* pTTTA = (TOOLTIPTEXTA*)pNMHDR;
|
|
|
+ TOOLTIPTEXTW* pTTTW = (TOOLTIPTEXTW*)pNMHDR;
|
|
|
+ CString strTipText;
|
|
|
+
|
|
|
+ UINT_PTR nID = pNMHDR->idFrom;
|
|
|
+
|
|
|
+ if(nID == 0) // Notification in NT from automatically
|
|
|
+ return FALSE; // created tooltip
|
|
|
+
|
|
|
+ ::SendMessage(pNMHDR->hwndFrom, TTM_SETMAXTIPWIDTH, 0, 500);
|
|
|
+
|
|
|
+ // Use Item's name as the tool tip. Change this for something different.
|
|
|
+ // Like use its file size, etc.
|
|
|
+ GetToolTipText((int)nID-1, strTipText);
|
|
|
+
|
|
|
+ //Replace the tabs with spaces, the tooltip didn't like the \t s
|
|
|
+ strTipText.Replace(_T("\t"), _T(" "));
|
|
|
+
|
|
|
+ int nLength = strTipText.GetLength()+2;
|
|
|
+
|
|
|
+#ifndef _UNICODE
|
|
|
+ if (pNMHDR->code == TTN_NEEDTEXTA)
|
|
|
+ {
|
|
|
+ if(m_pchTip != NULL)
|
|
|
+ delete m_pchTip;
|
|
|
+
|
|
|
+ m_pchTip = new TCHAR[nLength];
|
|
|
+ lstrcpyn(m_pchTip, strTipText, nLength-1);
|
|
|
+ m_pchTip[nLength-1] = 0;
|
|
|
+ pTTTW->lpszText = (WCHAR*)m_pchTip;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ if(m_pwchTip != NULL)
|
|
|
+ delete m_pwchTip;
|
|
|
+
|
|
|
+ m_pwchTip = new WCHAR[nLength];
|
|
|
+ _mbstowcsz(m_pwchTip, strTipText, nLength-1);
|
|
|
+ m_pwchTip[nLength-1] = 0; // end of text
|
|
|
+ pTTTW->lpszText = (WCHAR*)m_pwchTip;
|
|
|
+ }
|
|
|
+#else
|
|
|
+ if(pNMHDR->code == TTN_NEEDTEXTA)
|
|
|
+ {
|
|
|
+ if(m_pchTip != NULL)
|
|
|
+ delete m_pchTip;
|
|
|
+
|
|
|
+ m_pchTip = new TCHAR[nLength];
|
|
|
+ STRNCPY(m_pchTip, strTipText, nLength-1);
|
|
|
+ m_pchTip[nLength-1] = 0; // end of text
|
|
|
+ pTTTW->lpszText = (LPTSTR)m_pchTip;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ if(m_pwchTip != NULL)
|
|
|
+ delete m_pwchTip;
|
|
|
+
|
|
|
+ m_pwchTip = new WCHAR[nLength];
|
|
|
+ lstrcpyn(m_pwchTip, strTipText, nLength-1);
|
|
|
+ m_pwchTip[nLength-1] = 0;
|
|
|
+ pTTTW->lpszText = (LPTSTR) m_pwchTip;
|
|
|
+ }
|
|
|
+#endif
|
|
|
+ *pResult = 0;
|
|
|
+
|
|
|
+ return TRUE; // message was handled
|
|
|
+}
|
|
|
+
|
|
|
+INT_PTR CQListCtrl::OnToolHitTest(CPoint point, TOOLINFO * pTI) const
|
|
|
+{
|
|
|
+ CRect rect;
|
|
|
+ GetClientRect(&rect);
|
|
|
+ if(rect.PtInRect(point))
|
|
|
+ {
|
|
|
+ if(GetItemCount())
|
|
|
+ {
|
|
|
+ int nTopIndex = GetTopIndex();
|
|
|
+ int nBottomIndex = nTopIndex + GetCountPerPage();
|
|
|
+ if(nBottomIndex > GetItemCount()) nBottomIndex = GetItemCount();
|
|
|
+ for(int nIndex = nTopIndex; nIndex <= nBottomIndex; nIndex++)
|
|
|
+ {
|
|
|
+ GetItemRect(nIndex, rect, LVIR_BOUNDS);
|
|
|
+ if(rect.PtInRect(point))
|
|
|
+ {
|
|
|
+ pTI->hwnd = m_hWnd;
|
|
|
+ pTI->uId = (UINT)(nIndex+1);
|
|
|
+ pTI->lpszText = LPSTR_TEXTCALLBACK;
|
|
|
+ pTI->rect = rect;
|
|
|
+ return pTI->uId;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ return -1;
|
|
|
+}
|
|
|
+
|
|
|
+int CQListCtrl::OnCreate(LPCREATESTRUCT lpCreateStruct)
|
|
|
+{
|
|
|
+ if (CListCtrl::OnCreate(lpCreateStruct) == -1)
|
|
|
+ return -1;
|
|
|
+
|
|
|
+ EnableToolTips();
|
|
|
+
|
|
|
+ m_pToolTip = new CToolTipEx;
|
|
|
+ m_pToolTip->Create(this);
|
|
|
+
|
|
|
+ m_pToolTip->SetNotifyWnd(GetParent());
|
|
|
+
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+BOOL CQListCtrl::PreTranslateMessage(MSG* pMsg)
|
|
|
+{
|
|
|
+ DWORD dID;
|
|
|
+ if(m_Accels.OnMsg(pMsg, dID))
|
|
|
+ {
|
|
|
+ switch(dID)
|
|
|
+ {
|
|
|
+ case COPY_BUFFER_HOT_KEY_1_ID:
|
|
|
+ PutSelectedItemOnDittoCopyBuffer(0);
|
|
|
+ break;
|
|
|
+ case COPY_BUFFER_HOT_KEY_2_ID:
|
|
|
+ PutSelectedItemOnDittoCopyBuffer(1);
|
|
|
+ break;
|
|
|
+ case COPY_BUFFER_HOT_KEY_3_ID:
|
|
|
+ PutSelectedItemOnDittoCopyBuffer(2);
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ GetParent()->SendMessage(NM_SELECT_DB_ID, dID, 0);
|
|
|
+ }
|
|
|
+
|
|
|
+ return TRUE;
|
|
|
+ }
|
|
|
+
|
|
|
+ if(m_pToolTip)
|
|
|
+ {
|
|
|
+ if(m_pToolTip->OnMsg(pMsg))
|
|
|
+ return TRUE;
|
|
|
+ }
|
|
|
+
|
|
|
+ switch(pMsg->message)
|
|
|
+ {
|
|
|
+ case WM_KEYDOWN:
|
|
|
+ if(HandleKeyDown(pMsg->wParam, pMsg->lParam))
|
|
|
+ return TRUE;
|
|
|
+
|
|
|
+ break; // end case WM_KEYDOWN
|
|
|
+
|
|
|
+ case WM_VSCROLL:
|
|
|
+ ASSERT(FALSE);
|
|
|
+ break;
|
|
|
+ } // end switch(pMsg->message)
|
|
|
+
|
|
|
+ return CListCtrl::PreTranslateMessage(pMsg);
|
|
|
+}
|
|
|
+
|
|
|
+BOOL CQListCtrl::HandleKeyDown(WPARAM wParam, LPARAM lParam)
|
|
|
+{
|
|
|
+ if(m_pToolTip)
|
|
|
+ {
|
|
|
+ MSG Msg;
|
|
|
+ Msg.lParam = lParam;
|
|
|
+ Msg.wParam = wParam;
|
|
|
+ Msg.message = WM_KEYDOWN;
|
|
|
+ if(m_pToolTip->OnMsg(&Msg))
|
|
|
+ return TRUE;
|
|
|
+ }
|
|
|
+
|
|
|
+ WPARAM vk = wParam;
|
|
|
+
|
|
|
+ // if a number key was pressed
|
|
|
+ if('0' <= vk && vk <= '9')
|
|
|
+ {
|
|
|
+ // if <Ctrl> is required but is absent, then break
|
|
|
+ if(g_Opt.m_bUseCtrlNumAccel && !(GetKeyState(VK_CONTROL) & 0x8000))
|
|
|
+ return FALSE;
|
|
|
+
|
|
|
+ int index = (int)vk - '0';
|
|
|
+ // '0' is actually 10 in the ditto window
|
|
|
+ if(index == 0)
|
|
|
+ index = 10;
|
|
|
+ // translate num 1-10 into the actual index (based upon m_bStartTop)
|
|
|
+ index = GetFirstTenIndex(index);
|
|
|
+ GetParent()->SendMessage(NM_SELECT_INDEX, index, 0);
|
|
|
+ return TRUE;
|
|
|
+ }
|
|
|
+
|
|
|
+ if(VK_NUMPAD0 <= vk && vk <= VK_NUMPAD9)
|
|
|
+ {
|
|
|
+ // if <Ctrl> is required but is absent, then break
|
|
|
+ if( g_Opt.m_bUseCtrlNumAccel && !(GetKeyState(VK_CONTROL) & 0x8000) )
|
|
|
+ return FALSE;
|
|
|
+
|
|
|
+ int index = (int)vk - VK_NUMPAD0;
|
|
|
+ // '0' is actually 10 in the ditto window
|
|
|
+ if(index == 0)
|
|
|
+ index = 10;
|
|
|
+
|
|
|
+ // translate num 1-10 into the actual index (based upon m_bStartTop)
|
|
|
+ index = GetFirstTenIndex(index);
|
|
|
+ GetParent()->SendMessage(NM_SELECT_INDEX, index, 0);
|
|
|
+ return TRUE;
|
|
|
+ }
|
|
|
+
|
|
|
+ switch( vk )
|
|
|
+ {
|
|
|
+ case 'X': // Ctrl-X = Cut (prepare for moving the items into a Group)
|
|
|
+ if(GetKeyState(VK_CONTROL) & 0x8000)
|
|
|
+ {
|
|
|
+ LoadCopyOrCutToClipboard();
|
|
|
+
|
|
|
+ theApp.IC_Cut(); // uses selection
|
|
|
+ return TRUE;
|
|
|
+ }
|
|
|
+ break;
|
|
|
+
|
|
|
+ case 'C': // Ctrl-C = Copy (prepare for copying the items into a Group)
|
|
|
+ if(GetKeyState(VK_CONTROL) & 0x8000)
|
|
|
+ {
|
|
|
+ LoadCopyOrCutToClipboard();
|
|
|
+
|
|
|
+ theApp.IC_Copy(); // uses selection
|
|
|
+ return TRUE;
|
|
|
+ }
|
|
|
+ break;
|
|
|
+
|
|
|
+ case 'V': // Ctrl-V = Paste (actually performs the copy or move of items into the current Group)
|
|
|
+ if(GetKeyState(VK_CONTROL) & 0x8000)
|
|
|
+ {
|
|
|
+ theApp.IC_Paste();
|
|
|
+ return TRUE;
|
|
|
+ }
|
|
|
+ break;
|
|
|
+
|
|
|
+ case 'A': // Ctrl-A = Select All
|
|
|
+ if(GetKeyState(VK_CONTROL) & 0x8000)
|
|
|
+ {
|
|
|
+ int nCount = GetItemCount();
|
|
|
+ for(int i = 0; i < nCount; i++)
|
|
|
+ {
|
|
|
+ SetSelection(i);
|
|
|
+ }
|
|
|
+ return TRUE;
|
|
|
+ }
|
|
|
+ break;
|
|
|
+
|
|
|
+ case VK_F3:
|
|
|
+ {
|
|
|
+ ShowFullDescription();
|
|
|
+ return TRUE;
|
|
|
+ }
|
|
|
+ case VK_BACK:
|
|
|
+ theApp.EnterGroupID( theApp.m_GroupParentID );
|
|
|
+ return TRUE;
|
|
|
+ case VK_SPACE:
|
|
|
+ if(GetKeyState(VK_CONTROL) & 0x8000)
|
|
|
+ {
|
|
|
+ theApp.ShowPersistent( !g_Opt.m_bShowPersistent );
|
|
|
+ return TRUE;
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ } // end switch(vk)
|
|
|
+
|
|
|
+ return FALSE;
|
|
|
+}
|
|
|
+
|
|
|
+void CQListCtrl::LoadCopyOrCutToClipboard()
|
|
|
+{
|
|
|
+ ARRAY arr;
|
|
|
+ GetSelectionItemData(arr);
|
|
|
+ INT_PTR count = arr.GetSize();
|
|
|
+ if(count <= 0)
|
|
|
+ return;
|
|
|
+
|
|
|
+ CProcessPaste paste;
|
|
|
+
|
|
|
+ //Don't send the paste just load it into memory
|
|
|
+ paste.m_bSendPaste = false;
|
|
|
+
|
|
|
+ if(count > 1)
|
|
|
+ paste.GetClipIDs().Copy(arr);
|
|
|
+ else
|
|
|
+ paste.GetClipIDs().Add(arr[0]);
|
|
|
+
|
|
|
+ //Don't move these to the top
|
|
|
+ BOOL bItWas = g_Opt.m_bUpdateTimeOnPaste;
|
|
|
+ g_Opt.m_bUpdateTimeOnPaste = FALSE;
|
|
|
+
|
|
|
+ paste.DoPaste();
|
|
|
+
|
|
|
+ g_Opt.m_bUpdateTimeOnPaste = bItWas;
|
|
|
+}
|
|
|
+
|
|
|
+void CQListCtrl::ShowFullDescription(bool bFromAuto)
|
|
|
+{
|
|
|
+ int nItem = GetCaret();
|
|
|
+ CRect rc, crWindow;
|
|
|
+ GetWindowRect(&crWindow);
|
|
|
+ GetItemRect(nItem, rc, LVIR_BOUNDS);
|
|
|
+ ClientToScreen(rc);
|
|
|
+
|
|
|
+ CPoint pt;
|
|
|
+
|
|
|
+ if(bFromAuto == false)
|
|
|
+ {
|
|
|
+ pt = CPoint(rc.left, rc.bottom);
|
|
|
+ }
|
|
|
+ else
|
|
|
+ pt = CPoint((crWindow.left + (crWindow.right - crWindow.left)/2), rc.bottom);
|
|
|
+
|
|
|
+ CString csDescription;
|
|
|
+ GetToolTipText(nItem, csDescription);
|
|
|
+
|
|
|
+ m_pToolTip->DestroyWindow();
|
|
|
+
|
|
|
+ m_pToolTip = new CToolTipEx;
|
|
|
+ m_pToolTip->Create(this);
|
|
|
+ m_pToolTip->SetNotifyWnd(GetParent());
|
|
|
+
|
|
|
+ if(m_pToolTip)
|
|
|
+ {
|
|
|
+ m_pToolTip->SetToolTipText(_T(""));
|
|
|
+ m_pToolTip->SetRTFText(" ");
|
|
|
+ bool bSetPlainText = false;
|
|
|
+ CClipFormat Clip;
|
|
|
+
|
|
|
+ Clip.m_cfType = CF_UNICODETEXT;
|
|
|
+ if(GetClipData(nItem, Clip) && Clip.m_hgData)
|
|
|
+ {
|
|
|
+ LPVOID pvData = GlobalLock(Clip.m_hgData);
|
|
|
+ if(pvData)
|
|
|
+ {
|
|
|
+ CString csText = (WCHAR*)pvData;
|
|
|
+ m_pToolTip->SetToolTipText(csText);
|
|
|
+ bSetPlainText = true;
|
|
|
+ }
|
|
|
+
|
|
|
+ GlobalUnlock(Clip.m_hgData);
|
|
|
+
|
|
|
+ Clip.Free();
|
|
|
+ Clip.Clear();
|
|
|
+ }
|
|
|
+
|
|
|
+ if(bSetPlainText == false)
|
|
|
+ {
|
|
|
+ Clip.m_cfType = CF_TEXT;
|
|
|
+ if(GetClipData(nItem, Clip) && Clip.m_hgData)
|
|
|
+ {
|
|
|
+ LPVOID pvData = GlobalLock(Clip.m_hgData);
|
|
|
+ if(pvData)
|
|
|
+ {
|
|
|
+ CString csText = (char*)pvData;
|
|
|
+ m_pToolTip->SetToolTipText(csText);
|
|
|
+
|
|
|
+ bSetPlainText = true;
|
|
|
+ }
|
|
|
+
|
|
|
+ GlobalUnlock(Clip.m_hgData);
|
|
|
+
|
|
|
+ Clip.Free();
|
|
|
+ Clip.Clear();
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ if(bSetPlainText == false)
|
|
|
+ {
|
|
|
+ m_pToolTip->SetToolTipText(csDescription);
|
|
|
+ }
|
|
|
+
|
|
|
+ Clip.m_cfType = RegisterClipboardFormat(CF_RTF);
|
|
|
+
|
|
|
+ if(GetClipData(nItem, Clip) && Clip.m_hgData)
|
|
|
+ {
|
|
|
+ LPVOID pvData = GlobalLock(Clip.m_hgData);
|
|
|
+ if(pvData)
|
|
|
+ {
|
|
|
+ m_pToolTip->SetRTFText((char*)pvData);
|
|
|
+ }
|
|
|
+
|
|
|
+ GlobalUnlock(Clip.m_hgData);
|
|
|
+
|
|
|
+ Clip.Free();
|
|
|
+ Clip.Clear();
|
|
|
+ }
|
|
|
+
|
|
|
+ Clip.m_cfType = CF_DIB;
|
|
|
+
|
|
|
+ if(GetClipData(nItem, Clip) && Clip.m_hgData)
|
|
|
+ {
|
|
|
+ CBitmap *pBitMap = new CBitmap;
|
|
|
+ if(pBitMap)
|
|
|
+ {
|
|
|
+ CRect rcItem;
|
|
|
+ GetWindowRect(rcItem);
|
|
|
+
|
|
|
+ CDC *pDC = GetDC();;
|
|
|
+
|
|
|
+ CBitmapHelper::GetCBitmap(&Clip, pDC, pBitMap, (rcItem.Width() * 2));
|
|
|
+
|
|
|
+ ReleaseDC(pDC);
|
|
|
+
|
|
|
+ //Tooltip wnd will release
|
|
|
+ m_pToolTip->SetBitmap(pBitMap);
|
|
|
+ }
|
|
|
+
|
|
|
+ Clip.Free();
|
|
|
+ Clip.Clear();
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ m_pToolTip->Show(pt);
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+void CQListCtrl::GetToolTipText(int nItem, CString &csText)
|
|
|
+{
|
|
|
+ CWnd* pParent=GetParent();
|
|
|
+ if(pParent && (pParent->GetSafeHwnd() != NULL))
|
|
|
+ {
|
|
|
+ CQListToolTipText info;
|
|
|
+ memset(&info, 0, sizeof(info));
|
|
|
+ info.hdr.code = NM_GETTOOLTIPTEXT;
|
|
|
+ info.hdr.hwndFrom = GetSafeHwnd();
|
|
|
+ info.hdr.idFrom = GetDlgCtrlID();
|
|
|
+ info.lItem = nItem;
|
|
|
+ //plus 100 for extra info - shortcut and such
|
|
|
+ info.cchTextMax = g_Opt.m_bDescTextSize + 100;
|
|
|
+ info.pszText = csText.GetBufferSetLength(info.cchTextMax);
|
|
|
+
|
|
|
+ pParent->SendMessage(WM_NOTIFY,(WPARAM)info.hdr.idFrom,(LPARAM)&info);
|
|
|
+
|
|
|
+ csText.ReleaseBuffer();
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+BOOL CQListCtrl::GetClipData(int nItem, CClipFormat &Clip)
|
|
|
+{
|
|
|
+ return theApp.GetClipData(GetItemData(nItem), Clip);
|
|
|
+}
|
|
|
+
|
|
|
+DWORD CQListCtrl::GetItemData(int nItem)
|
|
|
+{
|
|
|
+ if((GetStyle() & LVS_OWNERDATA))
|
|
|
+ {
|
|
|
+ CWnd* pParent=GetParent();
|
|
|
+ if(pParent && (pParent->GetSafeHwnd() != NULL))
|
|
|
+ {
|
|
|
+ LV_DISPINFO info;
|
|
|
+ memset(&info, 0, sizeof(info));
|
|
|
+ info.hdr.code = LVN_GETDISPINFO;
|
|
|
+ info.hdr.hwndFrom = GetSafeHwnd();
|
|
|
+ info.hdr.idFrom = GetDlgCtrlID();
|
|
|
+
|
|
|
+ info.item.iItem = nItem;
|
|
|
+ info.item.lParam = -1;
|
|
|
+ info.item.mask = LVIF_PARAM;
|
|
|
+
|
|
|
+ pParent->SendMessage(WM_NOTIFY,(WPARAM)info.hdr.idFrom,(LPARAM)&info);
|
|
|
+
|
|
|
+ return (DWORD)info.item.lParam;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ return (DWORD)CListCtrl::GetItemData(nItem);
|
|
|
+}
|
|
|
+
|
|
|
+CClipFormatQListCtrl* CQListCtrl::GetItem_CF_DIB_ClipFormat(int nItem)
|
|
|
+{
|
|
|
+ CClipFormatQListCtrl *format = NULL;
|
|
|
+
|
|
|
+ CWnd* pParent=GetParent();
|
|
|
+ if(pParent && (pParent->GetSafeHwnd() != NULL))
|
|
|
+ {
|
|
|
+ LV_DISPINFO info;
|
|
|
+ memset(&info, 0, sizeof(info));
|
|
|
+ info.hdr.code = LVN_GETDISPINFO;
|
|
|
+ info.hdr.hwndFrom = GetSafeHwnd();
|
|
|
+ info.hdr.idFrom = GetDlgCtrlID();
|
|
|
+
|
|
|
+ info.item.iItem = nItem;
|
|
|
+ info.item.lParam = NULL;
|
|
|
+ info.item.mask = LVIF_CF_DIB;
|
|
|
+
|
|
|
+ pParent->SendMessage(WM_NOTIFY,(WPARAM)info.hdr.idFrom,(LPARAM)&info);
|
|
|
+
|
|
|
+ if(info.item.lParam != NULL)
|
|
|
+ {
|
|
|
+ format = (CClipFormatQListCtrl *)info.item.lParam;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ return format;
|
|
|
+}
|
|
|
+
|
|
|
+CClipFormatQListCtrl* CQListCtrl::GetItem_CF_RTF_ClipFormat(int nItem)
|
|
|
+{
|
|
|
+ CClipFormatQListCtrl *format = NULL;
|
|
|
+
|
|
|
+ CWnd* pParent=GetParent();
|
|
|
+ if(pParent && (pParent->GetSafeHwnd() != NULL))
|
|
|
+ {
|
|
|
+ LV_DISPINFO info;
|
|
|
+ memset(&info, 0, sizeof(info));
|
|
|
+ info.hdr.code = LVN_GETDISPINFO;
|
|
|
+ info.hdr.hwndFrom = GetSafeHwnd();
|
|
|
+ info.hdr.idFrom = GetDlgCtrlID();
|
|
|
+
|
|
|
+ info.item.iItem = nItem;
|
|
|
+ info.item.lParam = NULL;
|
|
|
+ info.item.mask = LVIF_CF_RICHTEXT;
|
|
|
+
|
|
|
+ pParent->SendMessage(WM_NOTIFY, (WPARAM)info.hdr.idFrom, (LPARAM)&info);
|
|
|
+
|
|
|
+ if(info.item.lParam != NULL)
|
|
|
+ {
|
|
|
+ format = (CClipFormatQListCtrl *)info.item.lParam;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ return format;
|
|
|
+}
|
|
|
+
|
|
|
+void CQListCtrl::OnHScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar)
|
|
|
+{
|
|
|
+ CListCtrl::OnHScroll(nSBCode, nPos, pScrollBar);
|
|
|
+}
|
|
|
+
|
|
|
+void CQListCtrl::DestroyAndCreateAccelerator(BOOL bCreate, CppSQLite3DB &db)
|
|
|
+{
|
|
|
+ m_Accels.m_Map.RemoveAll();
|
|
|
+
|
|
|
+ if(bCreate)
|
|
|
+ {
|
|
|
+ CMainTableFunctions::LoadAcceleratorKeys(m_Accels, db);
|
|
|
+
|
|
|
+ LoadDittoCopyBufferHotkeys();
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+void CQListCtrl::LoadDittoCopyBufferHotkeys()
|
|
|
+{
|
|
|
+ CCopyBufferItem Item;
|
|
|
+ CAccel a;
|
|
|
+
|
|
|
+ g_Opt.GetCopyBufferItem(0, Item);
|
|
|
+ if(Item.m_lCopyHotKey > 0)
|
|
|
+ {
|
|
|
+ a.Cmd = COPY_BUFFER_HOT_KEY_1_ID;
|
|
|
+ a.Key = Item.m_lCopyHotKey;
|
|
|
+ m_Accels.AddAccel(a);
|
|
|
+ }
|
|
|
+
|
|
|
+ g_Opt.GetCopyBufferItem(1, Item);
|
|
|
+ if(Item.m_lCopyHotKey > 0)
|
|
|
+ {
|
|
|
+ a.Cmd = COPY_BUFFER_HOT_KEY_2_ID;
|
|
|
+ a.Key = Item.m_lCopyHotKey;
|
|
|
+ m_Accels.AddAccel(a);
|
|
|
+ }
|
|
|
+
|
|
|
+ g_Opt.GetCopyBufferItem(2, Item);
|
|
|
+ if(Item.m_lCopyHotKey > 0)
|
|
|
+ {
|
|
|
+ a.Cmd = COPY_BUFFER_HOT_KEY_3_ID;
|
|
|
+ a.Key = Item.m_lCopyHotKey;
|
|
|
+ m_Accels.AddAccel(a);
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+void CQListCtrl::OnKillFocus(CWnd* pNewWnd)
|
|
|
+{
|
|
|
+ CListCtrl::OnKillFocus(pNewWnd);
|
|
|
+
|
|
|
+ //if(FocusOnToolTip() == FALSE)
|
|
|
+ //m_pToolTip->Hide();
|
|
|
+}
|
|
|
+
|
|
|
+HWND CQListCtrl::GetToolTipHWnd()
|
|
|
+{
|
|
|
+ return m_pToolTip->GetSafeHwnd();
|
|
|
+}
|
|
|
+
|
|
|
+BOOL CQListCtrl::SetItemCountEx(int iCount, DWORD dwFlags /* = 0 */)
|
|
|
+{
|
|
|
+ return CListCtrl::SetItemCountEx(iCount, dwFlags);
|
|
|
+}
|
|
|
+
|
|
|
+#define TIMER_SHOW_PROPERTIES 1
|
|
|
+
|
|
|
+void CQListCtrl::OnSelectionChange(NMHDR* pNMHDR, LRESULT* pResult)
|
|
|
+{
|
|
|
+ NMLISTVIEW *pnmv = (NMLISTVIEW *) pNMHDR;
|
|
|
+
|
|
|
+ if((pnmv->uNewState == 3) ||
|
|
|
+ (pnmv->uNewState == 1))
|
|
|
+ {
|
|
|
+ if(g_Opt.m_bAllwaysShowDescription)
|
|
|
+ {
|
|
|
+ KillTimer(TIMER_SHOW_PROPERTIES);
|
|
|
+ SetTimer(TIMER_SHOW_PROPERTIES, 300, NULL);
|
|
|
+ }
|
|
|
+ if(GetSelectedCount() > 0 )
|
|
|
+ theApp.SetStatus(NULL, FALSE);
|
|
|
+ }
|
|
|
+
|
|
|
+ if(GetSelectedCount() == this->GetItemCount())
|
|
|
+ {
|
|
|
+ if(m_allSelected == false)
|
|
|
+ {
|
|
|
+ Log(StrF(_T("List box Select All")));
|
|
|
+
|
|
|
+ GetParent()->SendMessage(NM_ALL_SELECTED, 0, 0);
|
|
|
+ m_allSelected = true;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else if(m_allSelected == true)
|
|
|
+ {
|
|
|
+ Log(StrF(_T("List box REMOVED Select All")));
|
|
|
+ m_allSelected = false;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+void CQListCtrl::OnTimer(UINT_PTR nIDEvent)
|
|
|
+{
|
|
|
+ if(nIDEvent == TIMER_SHOW_PROPERTIES)
|
|
|
+ {
|
|
|
+ if( theApp.m_bShowingQuickPaste )
|
|
|
+ ShowFullDescription(true);
|
|
|
+ KillTimer(TIMER_SHOW_PROPERTIES);
|
|
|
+ }
|
|
|
+
|
|
|
+ CListCtrl::OnTimer(nIDEvent);
|
|
|
+}
|
|
|
+
|
|
|
+void CQListCtrl::SetLogFont(LOGFONT &font)
|
|
|
+{
|
|
|
+ m_Font.DeleteObject();
|
|
|
+
|
|
|
+ m_Font.CreateFontIndirect(&font);
|
|
|
+
|
|
|
+ SetFont(&m_Font);
|
|
|
+}
|
|
|
+
|
|
|
+void CQListCtrl::OnVScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar)
|
|
|
+{
|
|
|
+ CListCtrl::OnVScroll(nSBCode, nPos, pScrollBar);
|
|
|
+}
|
|
|
+
|
|
|
+BOOL CQListCtrl::OnMouseWheel(UINT nFlags, short zDelta, CPoint pt)
|
|
|
+{
|
|
|
+ return CListCtrl::OnMouseWheel(nFlags, zDelta, pt);
|
|
|
+}
|
|
|
+
|
|
|
+BOOL CQListCtrl::OnChildNotify(UINT message, WPARAM wParam, LPARAM lParam, LRESULT* pLResult)
|
|
|
+{
|
|
|
+ NMLVCACHEHINT* pcachehint = NULL;
|
|
|
+
|
|
|
+ if(message == WM_NOTIFY)
|
|
|
+ {
|
|
|
+ NMHDR* phdr = (NMHDR*)lParam;
|
|
|
+
|
|
|
+ switch(phdr->code)
|
|
|
+ {
|
|
|
+ case LVN_ODCACHEHINT:
|
|
|
+ pcachehint= (NMLVCACHEHINT*) phdr;
|
|
|
+
|
|
|
+ GetParent()->SendMessage(NM_FILL_REST_OF_LIST, pcachehint->iFrom, pcachehint->iTo);
|
|
|
+ return FALSE;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ return CListCtrl::OnChildNotify(message, wParam, lParam, pLResult);
|
|
|
+}
|
|
|
+
|
|
|
+BOOL CQListCtrl::OnItemDeleted(long lID)
|
|
|
+{
|
|
|
+ BOOL bRet2 = m_RTFData.RemoveKey(lID);
|
|
|
+
|
|
|
+ return (bRet2);
|
|
|
}
|