/* * PROJECT: ReactOS Applications Manager * LICENSE: GPL-2.0-or-later (https://spdx.org/licenses/GPL-2.0-or-later) * PURPOSE: Application view class and other classes used by it * COPYRIGHT: Copyright 2020 He Yang (1160386205@qq.com) * Copyright 2022,2023 Mark Jansen */ #include "rapps.h" #include "appview.h" #include "gui.h" #include using namespace Gdiplus; HICON g_hDefaultPackageIcon = NULL; static int g_DefaultPackageIconILIdx = I_IMAGENONE; // **** Menu helpers **** BOOL DeleteMenuEx( _In_ HMENU hMenu, _In_ UINT uPosition, _In_ UINT uFlags) { INT pos; MENUITEMINFOW mii = { sizeof(mii), MIIM_FTYPE, 0 }; bool bIsValidItem1, bIsValidItem2; bool bIsSep1, bIsSep2; if (uFlags & MF_BYPOSITION) pos = (INT)uPosition; else pos = ::GetMenuPosFromID(hMenu, uPosition); if (pos < 0) return FALSE; bIsValidItem1 = ((pos > 0) && ::GetMenuItemInfoW(hMenu, pos - 1, TRUE, &mii)); bIsSep1 = bIsValidItem1 && !!(mii.fType & MFT_SEPARATOR); bIsValidItem2 = ::GetMenuItemInfoW(hMenu, pos + 1, TRUE, &mii); bIsSep2 = bIsValidItem2 && !!(mii.fType & MFT_SEPARATOR); if (bIsSep1 && !bIsSep2 && !bIsValidItem2) pos = pos - 1; // Delete separator only if pos+1 has no item else if (!bIsSep1 && bIsSep2 && !bIsValidItem1) pos = pos + 1; // Delete separator only if pos-1 has no item else if (bIsSep1 && bIsSep2) pos = pos + 1; else pos = -1; // Delete one of the separators if necessary if (pos != -1) ::DeleteMenu(hMenu, pos, MF_BYPOSITION); // Finally, delete the menu item itself. return ::DeleteMenu(hMenu, uPosition, uFlags); } // **** Menu helpers **** // **** CMainToolbar **** VOID CMainToolbar::AddImageToImageList(HIMAGELIST hImageList, UINT ImageIndex) { HICON hImage; if (!(hImage = (HICON)LoadImageW(hInst, MAKEINTRESOURCE(ImageIndex), IMAGE_ICON, m_iToolbarHeight, m_iToolbarHeight, 0))) { return; } ImageList_AddIcon(hImageList, hImage); DeleteObject(hImage); } HIMAGELIST CMainToolbar::InitImageList() { HIMAGELIST hImageList; /* Create the toolbar icon image list */ hImageList = ImageList_Create(m_iToolbarHeight, m_iToolbarHeight, ILC_MASK | GetSystemColorDepth(), 1, 1); if (!hImageList) { return NULL; } AddImageToImageList(hImageList, IDI_INSTALL); AddImageToImageList(hImageList, IDI_UNINSTALL); AddImageToImageList(hImageList, IDI_MODIFY); AddImageToImageList(hImageList, IDI_CHECK_ALL); AddImageToImageList(hImageList, IDI_REFRESH); AddImageToImageList(hImageList, IDI_UPDATE_DB); AddImageToImageList(hImageList, IDI_SETTINGS); AddImageToImageList(hImageList, IDI_EXIT); return hImageList; } CMainToolbar::CMainToolbar() : m_iToolbarHeight(24), m_dButtonsWidthMax(0) { } VOID CMainToolbar::OnGetDispInfo(LPTOOLTIPTEXT lpttt) { UINT idButton = (UINT)lpttt->hdr.idFrom; switch (idButton) { case ID_EXIT: lpttt->lpszText = MAKEINTRESOURCEW(IDS_TOOLTIP_EXIT); break; case ID_INSTALL: lpttt->lpszText = MAKEINTRESOURCEW(IDS_TOOLTIP_INSTALL); break; case ID_UNINSTALL: lpttt->lpszText = MAKEINTRESOURCEW(IDS_TOOLTIP_UNINSTALL); break; case ID_MODIFY: lpttt->lpszText = MAKEINTRESOURCEW(IDS_TOOLTIP_MODIFY); break; case ID_SETTINGS: lpttt->lpszText = MAKEINTRESOURCEW(IDS_TOOLTIP_SETTINGS); break; case ID_REFRESH: lpttt->lpszText = MAKEINTRESOURCEW(IDS_TOOLTIP_REFRESH); break; case ID_RESETDB: lpttt->lpszText = MAKEINTRESOURCEW(IDS_TOOLTIP_UPDATE_DB); break; } } HWND CMainToolbar::Create(HWND hwndParent) { CStringW szInstallBtn; CStringW szUninstallBtn; CStringW szModifyBtn; CStringW szSelectAllBtn; CStringW szRefreshBtn; CStringW szUpdateDbBtn; /* Load tooltip strings */ szInstallBtn.LoadStringW(IDS_TOOLTIP_INSTALL); szUninstallBtn.LoadStringW(IDS_TOOLTIP_UNINSTALL); szModifyBtn.LoadStringW(IDS_TOOLTIP_MODIFY); szSelectAllBtn.LoadStringW(IDS_TOOLTIP_SELECT_ALL); szRefreshBtn.LoadStringW(IDS_TOOLTIP_REFRESH); szUpdateDbBtn.LoadStringW(IDS_TOOLTIP_UPDATE_DB); /* Create buttons */ TBBUTTON Buttons[] = { /* iBitmap, idCommand, fsState, fsStyle, bReserved[2], dwData, iString */ {0, ID_INSTALL, TBSTATE_ENABLED, BTNS_BUTTON | BTNS_AUTOSIZE, {0}, 0, (INT_PTR)szInstallBtn.GetString()}, {1, ID_UNINSTALL, TBSTATE_ENABLED, BTNS_BUTTON | BTNS_AUTOSIZE, {0}, 0, (INT_PTR)szUninstallBtn.GetString()}, {2, ID_MODIFY, TBSTATE_ENABLED, BTNS_BUTTON | BTNS_AUTOSIZE, {0}, 0, (INT_PTR)szModifyBtn.GetString()}, {3, ID_CHECK_ALL, TBSTATE_ENABLED, BTNS_BUTTON | BTNS_AUTOSIZE, {0}, 0, (INT_PTR)szSelectAllBtn.GetString()}, {-1, 0, TBSTATE_ENABLED, BTNS_SEP, {0}, 0, 0}, {4, ID_REFRESH, TBSTATE_ENABLED, BTNS_BUTTON | BTNS_AUTOSIZE, {0}, 0, (INT_PTR)szRefreshBtn.GetString()}, {5, ID_RESETDB, TBSTATE_ENABLED, BTNS_BUTTON | BTNS_AUTOSIZE, {0}, 0, (INT_PTR)szUpdateDbBtn.GetString()}}; m_hWnd = CreateWindowExW( 0, TOOLBARCLASSNAMEW, NULL, WS_CHILD | WS_VISIBLE | TBSTYLE_FLAT | TBSTYLE_TOOLTIPS | TBSTYLE_LIST, 0, 0, 0, 0, hwndParent, 0, hInst, NULL); if (!m_hWnd) { return FALSE; } SendMessageW(TB_SETEXTENDEDSTYLE, 0, TBSTYLE_EX_HIDECLIPPEDBUTTONS); SetButtonStructSize(); /* Set image list */ HIMAGELIST hImageList = InitImageList(); if (hImageList) { ImageList_Destroy(SetImageList(hImageList)); } AddButtons(_countof(Buttons), Buttons); /* Remember the ideal width to use as a max width of buttons */ UpdateMaxButtonsWidth(); return m_hWnd; } void CMainToolbar::ShowButtonCaption(bool bShow) { DWORD dCurrentExStyle = (DWORD)SendMessageW(TB_GETEXTENDEDSTYLE, 0, 0); if (bShow) SendMessageW(TB_SETEXTENDEDSTYLE, 0, dCurrentExStyle & ~TBSTYLE_EX_MIXEDBUTTONS); else SendMessageW(TB_SETEXTENDEDSTYLE, 0, dCurrentExStyle | TBSTYLE_EX_MIXEDBUTTONS); } void CMainToolbar::UpdateMaxButtonsWidth() { SIZE size; GetIdealSize(FALSE, &size); m_dButtonsWidthMax = size.cx; } DWORD CMainToolbar::GetMaxButtonsWidth() const { return m_dButtonsWidthMax; } // **** CMainToolbar **** // **** CSearchBar **** CSearchBar::CSearchBar() : m_Width(180), m_Height(22) { } VOID CSearchBar::SetText(LPCWSTR lpszText) { SendMessageW(SB_SETTEXT, SBT_NOBORDERS, (LPARAM)lpszText); } HWND CSearchBar::Create(HWND hwndParent) { CStringW szBuf; m_hWnd = CreateWindowExW( WS_EX_CLIENTEDGE, L"Edit", NULL, WS_CHILD | WS_VISIBLE | ES_LEFT | ES_AUTOHSCROLL, 0, 0, m_Width, m_Height, hwndParent, (HMENU)NULL, hInst, 0); SendMessageW(WM_SETFONT, (WPARAM)GetStockObject(DEFAULT_GUI_FONT), 0); szBuf.LoadStringW(IDS_SEARCH_TEXT); SetWindowTextW(szBuf); return m_hWnd; } // **** CSearchBar **** // **** CComboBox **** CComboBox::CComboBox() : m_Width(80), m_Height(22) { } HWND CComboBox::Create(HWND hwndParent) { m_hWnd = CreateWindowW( WC_COMBOBOX, L"", CBS_DROPDOWNLIST | CBS_HASSTRINGS | WS_CHILD | WS_OVERLAPPED | WS_VISIBLE, 0, 0, m_Width, m_Height, hwndParent, NULL, 0, NULL); SendMessageW(WM_SETFONT, (WPARAM)GetStockObject(DEFAULT_GUI_FONT), 0); for (int i = 0; i < (int)_countof(m_TypeStringID); i++) { CStringW szBuf; szBuf.LoadStringW(m_TypeStringID[i]); SendMessageW(CB_ADDSTRING, 0, (LPARAM)(LPCWSTR)szBuf); } SendMessageW(CB_SETCURSEL, m_DefaultSelectType, 0); // select the first item return m_hWnd; } // **** CComboBox **** // **** CAppRichEdit **** VOID CAppRichEdit::LoadAndInsertText(UINT uStringID, const CStringW &szText, DWORD TextFlags) { CStringW szLoadedText; if (!szText.IsEmpty() && szLoadedText.LoadStringW(uStringID)) { const DWORD StringFlags = CFE_BOLD; InsertText(szLoadedText, StringFlags); InsertText(szText, TextFlags); } } VOID CAppRichEdit::LoadAndInsertText(UINT uStringID, DWORD StringFlags) { CStringW szLoadedText; if (szLoadedText.LoadStringW(uStringID)) { InsertText(L"\n", 0); InsertText(szLoadedText, StringFlags); InsertText(L"\n", 0); } } VOID CAppRichEdit::InsertTextWithString(UINT StringID, const CStringW &Text, DWORD TextFlags) { if (!Text.IsEmpty()) { LoadAndInsertText(StringID, Text, TextFlags); } } // **** CAppRichEdit **** int ScrnshotDownloadCallback(pASYNCINET AsyncInet, ASYNC_EVENT Event, WPARAM wParam, LPARAM lParam, VOID *Extension) { ScrnshotDownloadParam *DownloadParam = (ScrnshotDownloadParam *)Extension; switch (Event) { case ASYNCINET_DATA: DWORD BytesWritten; WriteFile(DownloadParam->hFile, (LPCVOID)wParam, (DWORD)lParam, &BytesWritten, NULL); break; case ASYNCINET_COMPLETE: CloseHandle(DownloadParam->hFile); SendMessage( DownloadParam->hwndNotify, WM_RAPPS_DOWNLOAD_COMPLETE, (WPARAM)ERROR_SUCCESS, (LPARAM)DownloadParam); break; case ASYNCINET_CANCELLED: CloseHandle(DownloadParam->hFile); SendMessage( DownloadParam->hwndNotify, WM_RAPPS_DOWNLOAD_COMPLETE, (WPARAM)ERROR_CANCELLED, (LPARAM)DownloadParam); break; case ASYNCINET_ERROR: CloseHandle(DownloadParam->hFile); SendMessage(DownloadParam->hwndNotify, WM_RAPPS_DOWNLOAD_COMPLETE, wParam, (LPARAM)DownloadParam); break; default: ATLASSERT(FALSE); break; } return 0; } // **** CAppScrnshotPreview **** CAppScrnshotPreview::CAppScrnshotPreview(const CStringW &BasePath) : m_BasePath(BasePath) { } BOOL CAppScrnshotPreview::ProcessWindowMessage( HWND hwnd, UINT Msg, WPARAM wParam, LPARAM lParam, LRESULT &theResult, DWORD dwMapId) { theResult = 0; switch (Msg) { case WM_CREATE: hBrokenImgIcon = (HICON)LoadImage(hInst, MAKEINTRESOURCE(IDI_BROKEN_IMAGE), IMAGE_ICON, BrokenImgSize, BrokenImgSize, 0); break; case WM_SIZE: { if (BrokenImgSize != min(min(GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam)), BROKENIMG_ICON_SIZE)) { BrokenImgSize = min(min(GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam)), BROKENIMG_ICON_SIZE); if (hBrokenImgIcon) { DeleteObject(hBrokenImgIcon); hBrokenImgIcon = (HICON)LoadImage( hInst, MAKEINTRESOURCE(IDI_BROKEN_IMAGE), IMAGE_ICON, BrokenImgSize, BrokenImgSize, 0); } } break; } case WM_RAPPS_DOWNLOAD_COMPLETE: { ScrnshotDownloadParam *DownloadParam = (ScrnshotDownloadParam *)lParam; AsyncInetRelease(AsyncInet); AsyncInet = NULL; switch (wParam) { case ERROR_SUCCESS: if (ContentID == DownloadParam->ID) { DisplayFile(DownloadParam->DownloadFileName); // send a message to trigger resizing ::SendMessageW(::GetParent(m_hWnd), WM_RAPPS_RESIZE_CHILDREN, 0, 0); InvalidateRect(0, 0); TempImagePath = DownloadParam->DownloadFileName; // record tmp file path in order to delete it when cleanup } else { // the picture downloaded is already outdated. delete it. DeleteFileW(DownloadParam->DownloadFileName); } break; case ERROR_CANCELLED: DeleteFileW(DownloadParam->DownloadFileName); break; default: DisplayFailed(); // send a message to trigger resizing ::SendMessageW(::GetParent(m_hWnd), WM_RAPPS_RESIZE_CHILDREN, 0, 0); InvalidateRect(0, 0); DeleteFileW(DownloadParam->DownloadFileName); break; } delete DownloadParam; break; } case WM_PAINT: { PAINTSTRUCT ps; HDC hdc = BeginPaint(&ps); CRect rect; GetClientRect(&rect); PaintOnDC(hdc, rect.Width(), rect.Height(), ps.fErase); EndPaint(&ps); break; } case WM_PRINTCLIENT: { if (lParam & PRF_CHECKVISIBLE) { if (!IsWindowVisible()) break; } CRect rect; GetClientRect(&rect); PaintOnDC((HDC)wParam, rect.Width(), rect.Height(), lParam & PRF_ERASEBKGND); break; } case WM_ERASEBKGND: { return TRUE; // do not erase to avoid blinking } case WM_TIMER: { switch (wParam) { case TIMER_LOADING_ANIMATION: LoadingAnimationFrame++; LoadingAnimationFrame %= (LOADING_ANIMATION_PERIOD * LOADING_ANIMATION_FPS); HDC hdc = GetDC(); CRect rect; GetClientRect(&rect); PaintOnDC(hdc, rect.Width(), rect.Height(), TRUE); ReleaseDC(hdc); } break; } case WM_DESTROY: { PreviousDisplayCleanup(); DeleteObject(hBrokenImgIcon); hBrokenImgIcon = NULL; break; } } return FALSE; } VOID CAppScrnshotPreview::DisplayLoading() { SetStatus(SCRNSHOT_PREV_LOADING); if (bLoadingTimerOn) { KillTimer(TIMER_LOADING_ANIMATION); } LoadingAnimationFrame = 0; bLoadingTimerOn = TRUE; SetTimer(TIMER_LOADING_ANIMATION, 1000 / LOADING_ANIMATION_FPS, 0); } VOID CAppScrnshotPreview::DisplayFailed() { InterlockedIncrement64(&ContentID); SetStatus(SCRNSHOT_PREV_FAILED); PreviousDisplayCleanup(); } BOOL CAppScrnshotPreview::DisplayFile(LPCWSTR lpszFileName) { PreviousDisplayCleanup(); SetStatus(SCRNSHOT_PREV_IMAGE); pImage = Bitmap::FromFile(lpszFileName, 0); if (pImage->GetLastStatus() != Ok) { DisplayFailed(); return FALSE; } return TRUE; } VOID CAppScrnshotPreview::SetStatus(SCRNSHOT_STATUS Status) { ScrnshotPrevStauts = Status; } VOID CAppScrnshotPreview::PaintOnDC(HDC hdc, int width, int height, BOOL bDrawBkgnd) { // use an off screen dc to avoid blinking HDC hdcMem = CreateCompatibleDC(hdc); HBITMAP hBitmap = CreateCompatibleBitmap(hdc, width, height); SelectObject(hdcMem, hBitmap); if (bDrawBkgnd) { HBRUSH hOldBrush = (HBRUSH)SelectObject(hdcMem, (HGDIOBJ)GetSysColorBrush(COLOR_BTNFACE)); PatBlt(hdcMem, 0, 0, width, height, PATCOPY); SelectObject(hdcMem, hOldBrush); } switch (ScrnshotPrevStauts) { case SCRNSHOT_PREV_EMPTY: { } break; case SCRNSHOT_PREV_LOADING: { Graphics graphics(hdcMem); Color color(255, 0, 0); SolidBrush dotBrush(Color(255, 100, 100, 100)); graphics.SetSmoothingMode(SmoothingMode::SmoothingModeAntiAlias); // Paint three dot float DotWidth = GetLoadingDotWidth(width, height); graphics.FillEllipse( (Brush *)(&dotBrush), (REAL)width / 2.0 - min(width, height) * 2.0 / 16.0 - DotWidth / 2.0, (REAL)height / 2.0 - GetFrameDotShift(LoadingAnimationFrame + LOADING_ANIMATION_FPS / 4, width, height) - DotWidth / 2.0, DotWidth, DotWidth); graphics.FillEllipse( (Brush *)(&dotBrush), (REAL)width / 2.0 - DotWidth / 2.0, (REAL)height / 2.0 - GetFrameDotShift(LoadingAnimationFrame, width, height) - DotWidth / 2.0, DotWidth, DotWidth); graphics.FillEllipse( (Brush *)(&dotBrush), (REAL)width / 2.0 + min(width, height) * 2.0 / 16.0 - DotWidth / 2.0, (REAL)height / 2.0 - GetFrameDotShift(LoadingAnimationFrame - LOADING_ANIMATION_FPS / 4, width, height) - DotWidth / 2.0, DotWidth, DotWidth); } break; case SCRNSHOT_PREV_IMAGE: { if (pImage) { // always draw entire image inside the window. Graphics graphics(hdcMem); float ZoomRatio = min(((float)width / (float)pImage->GetWidth()), ((float)height / (float)pImage->GetHeight())); float ZoomedImgWidth = ZoomRatio * (float)pImage->GetWidth(); float ZoomedImgHeight = ZoomRatio * (float)pImage->GetHeight(); graphics.DrawImage( pImage, ((float)width - ZoomedImgWidth) / 2.0, ((float)height - ZoomedImgHeight) / 2.0, ZoomedImgWidth, ZoomedImgHeight); } } break; case SCRNSHOT_PREV_FAILED: { DrawIconEx( hdcMem, (width - BrokenImgSize) / 2, (height - BrokenImgSize) / 2, hBrokenImgIcon, BrokenImgSize, BrokenImgSize, NULL, NULL, DI_NORMAL | DI_COMPAT); } break; } // copy the content form off-screen dc to hdc BitBlt(hdc, 0, 0, width, height, hdcMem, 0, 0, SRCCOPY); DeleteDC(hdcMem); DeleteObject(hBitmap); } float CAppScrnshotPreview::GetLoadingDotWidth(int width, int height) { return min(width, height) / 20.0; } float CAppScrnshotPreview::GetFrameDotShift(int Frame, int width, int height) { return min(width, height) * (1.0 / 16.0) * (2.0 / (2.0 - sqrt(3.0))) * (max(sin((float)Frame * 2 * PI / (LOADING_ANIMATION_PERIOD * LOADING_ANIMATION_FPS)), sqrt(3.0) / 2.0) - sqrt(3.0) / 2.0); } ATL::CWndClassInfo & CAppScrnshotPreview::GetWndClassInfo() { DWORD csStyle = CS_VREDRAW | CS_HREDRAW; static ATL::CWndClassInfo wc = { {sizeof(WNDCLASSEX), csStyle, StartWindowProc, 0, 0, NULL, 0, LoadCursorW(NULL, IDC_ARROW), (HBRUSH)(COLOR_BTNFACE + 1), 0, L"RAppsScrnshotPreview", NULL}, NULL, NULL, IDC_ARROW, TRUE, 0, _T("")}; return wc; } HWND CAppScrnshotPreview::Create(HWND hParent) { RECT r = {0, 0, 0, 0}; return CWindowImpl::Create(hParent, r, L"", WS_CHILD | WS_VISIBLE); } VOID CAppScrnshotPreview::PreviousDisplayCleanup() { if (bLoadingTimerOn) { KillTimer(TIMER_LOADING_ANIMATION); bLoadingTimerOn = FALSE; } LoadingAnimationFrame = 0; if (pImage) { delete pImage; pImage = NULL; } if (AsyncInet) { AsyncInetCancel(AsyncInet); } if (!TempImagePath.IsEmpty()) { DeleteFileW(TempImagePath.GetString()); TempImagePath.Empty(); } } VOID CAppScrnshotPreview::DisplayEmpty() { InterlockedIncrement64(&ContentID); SetStatus(SCRNSHOT_PREV_EMPTY); PreviousDisplayCleanup(); } BOOL CAppScrnshotPreview::DisplayImage(LPCWSTR lpszLocation) { LONGLONG ID = InterlockedIncrement64(&ContentID); PreviousDisplayCleanup(); if (PathIsURLW(lpszLocation)) { DisplayLoading(); ScrnshotDownloadParam *DownloadParam = new ScrnshotDownloadParam; if (!DownloadParam) return FALSE; DownloadParam->hwndNotify = m_hWnd; DownloadParam->ID = ID; // generate a filename CStringW ScrnshotFolder = m_BasePath; PathAppendW(ScrnshotFolder.GetBuffer(MAX_PATH), L"screenshots"); ScrnshotFolder.ReleaseBuffer(); if (!PathIsDirectoryW(ScrnshotFolder.GetString())) { CreateDirectoryW(ScrnshotFolder.GetString(), NULL); } if (!GetTempFileNameW( ScrnshotFolder.GetString(), L"img", 0, DownloadParam->DownloadFileName.GetBuffer(MAX_PATH))) { DownloadParam->DownloadFileName.ReleaseBuffer(); delete DownloadParam; DisplayFailed(); return FALSE; } DownloadParam->DownloadFileName.ReleaseBuffer(); DownloadParam->hFile = CreateFileW( DownloadParam->DownloadFileName.GetString(), GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL); if (DownloadParam->hFile == INVALID_HANDLE_VALUE) { delete DownloadParam; DisplayFailed(); return FALSE; } AsyncInet = AsyncInetDownload( 0, INTERNET_OPEN_TYPE_PRECONFIG, 0, 0, lpszLocation, TRUE, ScrnshotDownloadCallback, DownloadParam); if (!AsyncInet) { CloseHandle(DownloadParam->hFile); DeleteFileW(DownloadParam->DownloadFileName.GetBuffer()); delete DownloadParam; DisplayFailed(); return FALSE; } return TRUE; } else { return DisplayFile(lpszLocation); } } int CAppScrnshotPreview::GetRequestedWidth(int Height) // calculate requested window width by given height { switch (ScrnshotPrevStauts) { case SCRNSHOT_PREV_EMPTY: return 0; case SCRNSHOT_PREV_LOADING: return 200; case SCRNSHOT_PREV_IMAGE: if (pImage) { // return the width needed to display image inside the window. // and always keep window w/h ratio inside [ 1/SCRNSHOT_MAX_ASPECT_RAT, SCRNSHOT_MAX_ASPECT_RAT ] return (int)floor( (float)Height * max(min((float)pImage->GetWidth() / (float)pImage->GetHeight(), (float)SCRNSHOT_MAX_ASPECT_RAT), 1.0 / (float)SCRNSHOT_MAX_ASPECT_RAT)); } return 0; case SCRNSHOT_PREV_FAILED: return 200; default: return 0; } } CAppScrnshotPreview::~CAppScrnshotPreview() { PreviousDisplayCleanup(); } // **** CAppScrnshotPreview **** // **** CAppInfoDisplay **** BOOL CAppInfoDisplay::ProcessWindowMessage( HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam, LRESULT &theResult, DWORD dwMapId) { theResult = 0; switch (message) { case WM_CREATE: { RichEdit = new CAppRichEdit(); RichEdit->Create(hwnd); CStringW Directory; ::GetStorageDirectory(Directory); ScrnshotPrev = new CAppScrnshotPreview(Directory); ScrnshotPrev->Create(hwnd); break; } case WM_SIZE: { ResizeChildren(GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam)); break; } case WM_RAPPS_RESIZE_CHILDREN: { ResizeChildren(); break; } case WM_COMMAND: { OnCommand(wParam, lParam); break; } case WM_NOTIFY: { NMHDR *NotifyHeader = (NMHDR *)lParam; if (NotifyHeader->hwndFrom == RichEdit->m_hWnd) { switch (NotifyHeader->code) { case EN_LINK: OnLink((ENLINK *)lParam); break; } } break; } case WM_SYSCOLORCHANGE: { RichEdit->SendMessageW(EM_SETBKGNDCOLOR, 0, GetSysColor(COLOR_BTNFACE)); break; } } return FALSE; } VOID CAppInfoDisplay::ResizeChildren() { CRect rect; GetWindowRect(&rect); ResizeChildren(rect.Width(), rect.Height()); } VOID CAppInfoDisplay::ResizeChildren(int Width, int Height) { int ScrnshotWidth = ScrnshotPrev->GetRequestedWidth(Height); // make sure richedit always have room to display ScrnshotWidth = min(ScrnshotWidth, Width - INFO_DISPLAY_PADDING - RICHEDIT_MIN_WIDTH); DWORD dwError = ERROR_SUCCESS; HDWP hDwp = BeginDeferWindowPos(2); if (hDwp) { hDwp = ::DeferWindowPos(hDwp, ScrnshotPrev->m_hWnd, NULL, 0, 0, ScrnshotWidth, Height, 0); if (hDwp) { // hide the padding if scrnshot window width == 0 int RicheditPosX = ScrnshotWidth ? (ScrnshotWidth + INFO_DISPLAY_PADDING) : 0; hDwp = ::DeferWindowPos(hDwp, RichEdit->m_hWnd, NULL, RicheditPosX, 0, Width - RicheditPosX, Height, 0); if (hDwp) { EndDeferWindowPos(hDwp); } else { dwError = GetLastError(); } } else { dwError = GetLastError(); } } else { dwError = GetLastError(); } #if DBG ATLASSERT(dwError == ERROR_SUCCESS); #endif UNREFERENCED_PARAMETER(dwError); UpdateWindow(); } VOID CAppInfoDisplay::OnLink(ENLINK *Link) { switch (Link->msg) { case WM_LBUTTONUP: case WM_RBUTTONUP: { if (pLink) HeapFree(GetProcessHeap(), 0, pLink); pLink = (LPWSTR)HeapAlloc( GetProcessHeap(), 0, (max(Link->chrg.cpMin, Link->chrg.cpMax) - min(Link->chrg.cpMin, Link->chrg.cpMax) + 1) * sizeof(WCHAR)); if (!pLink) { /* TODO: Error message */ return; } RichEdit->SendMessageW(EM_SETSEL, Link->chrg.cpMin, Link->chrg.cpMax); RichEdit->SendMessageW(EM_GETSELTEXT, 0, (LPARAM)pLink); ShowPopupMenuEx(m_hWnd, m_hWnd, IDR_LINKMENU, -1); } break; } } ATL::CWndClassInfo & CAppInfoDisplay::GetWndClassInfo() { DWORD csStyle = CS_VREDRAW | CS_HREDRAW; static ATL::CWndClassInfo wc = {/*.m_wc=*/ {/*cbSize=*/sizeof(WNDCLASSEX), /*style=*/csStyle, /*lpfnWndProc=*/StartWindowProc, /*cbClsExtra=*/0, /*cbWndExtra=*/0, /*hInstance=*/NULL, /*hIcon=*/NULL, /*hCursor*/ NULL, /*hbrBackground=*/(HBRUSH)(COLOR_BTNFACE + 1), /*lpszMenuName=*/NULL, /*lpszClassName=*/L"RAppsAppInfo", /*hIconSm=*/NULL}, /*m_lpszOrigName=*/NULL, /*pWndProc=*/NULL, /*m_lpszCursorID=*/IDC_ARROW, /*m_bSystemCursor*/ TRUE, /*m_atom=*/0, /*m_szAutoName=*/_T("")}; return wc; } HWND CAppInfoDisplay::Create(HWND hwndParent) { RECT r = {0, 0, 0, 0}; return CWindowImpl::Create(hwndParent, r, L"", WS_CHILD | WS_VISIBLE | WS_CLIPCHILDREN | WS_CLIPSIBLINGS); } VOID CAppInfoDisplay::ShowAppInfo(CAppInfo *Info) { CStringW ScrnshotLocation; if (Info->RetrieveScreenshot(ScrnshotLocation)) { ScrnshotPrev->DisplayImage(ScrnshotLocation); } else { ScrnshotPrev->DisplayEmpty(); } ResizeChildren(); Info->ShowAppInfo(RichEdit); } void CAppInfoDisplay::SetWelcomeText(bool bAppwiz) { CStringW szText; ScrnshotPrev->DisplayEmpty(); ResizeChildren(); // Display the standard banner in normal mode, or // the specific "Add/Remove Programs" in APPWIZ-mode. if (!bAppwiz) { szText.LoadStringW(IDS_WELCOME_TITLE); RichEdit->SetText(szText, CFE_BOLD); RichEdit->InsertText(L"\n\n", 0); szText.LoadStringW(IDS_WELCOME_TEXT); RichEdit->InsertText(szText, 0); szText.LoadStringW(IDS_WELCOME_URL); RichEdit->InsertText(szText, CFM_LINK); } else { szText.LoadStringW(IDS_APPWIZ_TITLE); RichEdit->SetText(szText, CFE_BOLD); RichEdit->InsertText(L"\n\n", 0); szText.LoadStringW(IDS_APPWIZ_TEXT1); RichEdit->InsertText(szText, 0); RichEdit->InsertText(L"\n", 0); szText.LoadStringW(IDS_APPWIZ_TEXT2); RichEdit->InsertText(szText, 0); } } VOID CAppInfoDisplay::OnCommand(WPARAM wParam, LPARAM lParam) { WORD wCommand = LOWORD(wParam); switch (wCommand) { case ID_OPEN_LINK: ShellExecuteW(m_hWnd, L"open", pLink, NULL, NULL, SW_SHOWNOACTIVATE); HeapFree(GetProcessHeap(), 0, pLink); pLink = NULL; break; case ID_COPY_LINK: CopyTextToClipboard(pLink); HeapFree(GetProcessHeap(), 0, pLink); pLink = NULL; break; } } CAppInfoDisplay::~CAppInfoDisplay() { delete RichEdit; delete ScrnshotPrev; } // **** CAppInfoDisplay **** // **** CAppsListView **** struct CAsyncLoadIcon { CAsyncLoadIcon *pNext; HWND hAppsList; CAppInfo *AppInfo; // Only used to find the item in the list, do not access on background thread UINT TaskId; bool Parse; WCHAR Location[ANYSIZE_ARRAY]; void Free() { free(this); } static CAsyncLoadIcon* Queue(HWND hAppsList, CAppInfo &AppInfo, bool Parse); static void StartTasks(); } *g_AsyncIconTasks = NULL; static UINT g_AsyncIconTaskId = 0; static DWORD CALLBACK AsyncLoadIconProc(LPVOID Param) { for (CAsyncLoadIcon *task = (CAsyncLoadIcon*)Param, *old; task; old->Free()) { if (task->TaskId == g_AsyncIconTaskId) { HICON hIcon; if (!task->Parse) hIcon = (HICON)LoadImageW(NULL, task->Location, IMAGE_ICON, 0, 0, LR_LOADFROMFILE | LR_DEFAULTSIZE); else if (!ExtractIconExW(task->Location, PathParseIconLocationW(task->Location), &hIcon, NULL, 1)) hIcon = NULL; if (hIcon) { SendMessageW(task->hAppsList, WM_RAPPSLIST_ASYNCICON, (WPARAM)hIcon, (LPARAM)task); DestroyIcon(hIcon); } } old = task; task = task->pNext; } return 0; } CAsyncLoadIcon* CAsyncLoadIcon::Queue(HWND hAppsList, CAppInfo &AppInfo, bool Parse) { ATLASSERT(GetCurrentThreadId() == GetWindowThreadProcessId(hAppsList, NULL)); CStringW szIconPath; if (!AppInfo.RetrieveIcon(szIconPath)) return NULL; SIZE_T cbstr = (szIconPath.GetLength() + 1) * sizeof(WCHAR); CAsyncLoadIcon *task = (CAsyncLoadIcon*)malloc(sizeof(CAsyncLoadIcon) + cbstr); if (!task) return NULL; task->hAppsList = hAppsList; task->AppInfo = &AppInfo; task->TaskId = g_AsyncIconTaskId; task->Parse = Parse; CopyMemory(task->Location, szIconPath.GetBuffer(), cbstr); szIconPath.ReleaseBuffer(); task->pNext = g_AsyncIconTasks; g_AsyncIconTasks = task; return task; } void CAsyncLoadIcon::StartTasks() { CAsyncLoadIcon *tasks = g_AsyncIconTasks; g_AsyncIconTasks = NULL; if (HANDLE hThread = CreateThread(NULL, 0, AsyncLoadIconProc, tasks, 0, NULL)) CloseHandle(hThread); else AsyncLoadIconProc(tasks); // Insist so we at least free the tasks } CAppsListView::CAppsListView() { m_hImageListView = 0; } CAppsListView::~CAppsListView() { if (m_hImageListView) ImageList_Destroy(m_hImageListView); if (g_hDefaultPackageIcon) DestroyIcon(g_hDefaultPackageIcon); } LRESULT CAppsListView::OnEraseBackground(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled) { LRESULT lRes = this->DefWindowProc(uMsg, wParam, lParam); if (!m_Watermark.IsEmpty()) { RECT rc; GetClientRect(&rc); HGDIOBJ oldFont = SelectFont(HDC(wParam), GetStockFont(DEFAULT_GUI_FONT)); DrawShadowText( HDC(wParam), m_Watermark.GetString(), m_Watermark.GetLength(), &rc, DT_CENTER | DT_VCENTER | DT_NOPREFIX | DT_SINGLELINE, GetSysColor(COLOR_GRAYTEXT), GetSysColor(COLOR_GRAYTEXT), 1, 1); SelectFont(HDC(wParam), oldFont); } return lRes; } LRESULT CAppsListView::OnAsyncIcon(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled) { CAsyncLoadIcon *task = (CAsyncLoadIcon*)lParam; bHandled = TRUE; if (task->TaskId == g_AsyncIconTaskId) { LVITEM lvi; LVFINDINFO lvfi; lvfi.flags = LVFI_PARAM; lvfi.lParam = (LPARAM)task->AppInfo; lvi.iItem = ListView_FindItem(m_hWnd, -1, &lvfi); if (lvi.iItem != -1) { lvi.iImage = ImageList_AddIcon(m_hImageListView, (HICON)wParam); if (lvi.iImage != -1) { lvi.mask = LVIF_IMAGE; lvi.iSubItem = 0; ListView_SetItem(m_hWnd, &lvi); } } } return 0; } VOID CAppsListView::SetWatermark(const CStringW &Text) { m_Watermark = Text; } void CAppsListView::ShowCheckboxes(bool bShow) { SetExtendedListViewStyle((bShow ? LVS_EX_CHECKBOXES : 0) | LVS_EX_FULLROWSELECT); bHasCheckboxes = bShow; } VOID CAppsListView::ColumnClick(LPNMLISTVIEW pnmv) { HWND hHeader; HDITEMW hColumn; INT nHeaderID = pnmv->iSubItem; if ((GetWindowLongPtr(GWL_STYLE) & ~LVS_NOSORTHEADER) == 0) return; hHeader = (HWND)SendMessage(LVM_GETHEADER, 0, 0); ZeroMemory(&hColumn, sizeof(hColumn)); /* If the sorting column changed, remove the sorting style from the old column */ if ((nLastHeaderID != -1) && (nLastHeaderID != nHeaderID)) { bIsAscending = TRUE; // also reset sorting method to ascending hColumn.mask = HDI_FORMAT; Header_GetItem(hHeader, nLastHeaderID, &hColumn); hColumn.fmt &= ~(HDF_SORTUP | HDF_SORTDOWN); Header_SetItem(hHeader, nLastHeaderID, &hColumn); } /* Set the sorting style to the new column */ hColumn.mask = HDI_FORMAT; Header_GetItem(hHeader, nHeaderID, &hColumn); hColumn.fmt &= ~(HDF_SORTDOWN | HDF_SORTUP); hColumn.fmt |= (bIsAscending ? HDF_SORTUP : HDF_SORTDOWN); Header_SetItem(hHeader, nHeaderID, &hColumn); /* Sort the list, using the current values of nHeaderID and bIsAscending */ SortContext ctx = {this, nHeaderID}; SortItems(s_CompareFunc, &ctx); /* Save new values */ nLastHeaderID = nHeaderID; bIsAscending = !bIsAscending; } BOOL CAppsListView::AddColumn(INT Index, CStringW &Text, INT Width, INT Format) { LVCOLUMNW Column; ZeroMemory(&Column, sizeof(Column)); Column.mask = LVCF_FMT | LVCF_TEXT | LVCF_WIDTH | LVCF_SUBITEM; Column.iSubItem = Index; Column.pszText = const_cast(Text.GetString()); Column.cx = Width; Column.fmt = Format; return SendMessage(LVM_INSERTCOLUMN, Index, (LPARAM)(&Column)); } void CAppsListView::DeleteColumn(INT Index) { SendMessage(LVM_DELETECOLUMN, Index, 0); return; } INT CAppsListView::AddItem(INT ItemIndex, INT IconIndex, LPCWSTR lpText, LPARAM lParam) { LVITEMW Item; ZeroMemory(&Item, sizeof(Item)); Item.mask = LVIF_TEXT | LVIF_PARAM | LVIF_STATE; Item.pszText = const_cast(lpText); Item.lParam = lParam; Item.iItem = ItemIndex; Item.iImage = IconIndex; if (IconIndex >= 0) { Item.iImage = IconIndex; Item.mask |= LVIF_IMAGE; } return InsertItem(&Item); } HIMAGELIST CAppsListView::GetImageList(int iImageList) { return (HIMAGELIST)SendMessage(LVM_GETIMAGELIST, iImageList, 0); } INT CALLBACK CAppsListView::s_CompareFunc(LPARAM lParam1, LPARAM lParam2, LPARAM lParamSort) { SortContext *ctx = ((SortContext *)lParamSort); return ctx->lvw->CompareFunc(lParam1, lParam2, ctx->iSubItem); } INT CAppsListView::CompareFunc(LPARAM lParam1, LPARAM lParam2, INT iSubItem) { CStringW Item1, Item2; LVFINDINFOW IndexInfo; INT Index; IndexInfo.flags = LVFI_PARAM; IndexInfo.lParam = lParam1; Index = FindItem(-1, &IndexInfo); GetItemText(Index, iSubItem, Item1.GetBuffer(MAX_STR_LEN), MAX_STR_LEN); Item1.ReleaseBuffer(); IndexInfo.lParam = lParam2; Index = FindItem(-1, &IndexInfo); GetItemText(Index, iSubItem, Item2.GetBuffer(MAX_STR_LEN), MAX_STR_LEN); Item2.ReleaseBuffer(); return bIsAscending ? Item1.Compare(Item2) : Item2.Compare(Item1); } HWND CAppsListView::Create(HWND hwndParent) { RECT r = {205, 28, 465, 250}; DWORD style = WS_CHILD | WS_VISIBLE | LVS_SORTASCENDING | LVS_REPORT | LVS_SINGLESEL | LVS_SHOWSELALWAYS | LVS_AUTOARRANGE | LVS_SHAREIMAGELISTS; HWND hwnd = CListView::Create(hwndParent, r, NULL, style, WS_EX_CLIENTEDGE); if (hwnd) { ShowCheckboxes(false); } #pragma push_macro("SubclassWindow") #undef SubclassWindow m_hWnd = NULL; SubclassWindow(hwnd); #pragma pop_macro("SubclassWindow") return hwnd; } BOOL CAppsListView::GetCheckState(INT item) { return (BOOL)(GetItemState(item, LVIS_STATEIMAGEMASK) >> 12) - 1; } VOID CAppsListView::SetCheckState(INT item, BOOL fCheck) { if (bHasCheckboxes) { SetItemState(item, INDEXTOSTATEIMAGEMASK((fCheck) ? 2 : 1), LVIS_STATEIMAGEMASK); } } VOID CAppsListView::CheckAll() { if (bHasCheckboxes) { if (CheckedItemCount == ItemCount) { // clear all SetCheckState(-1, FALSE); } else { // check all SetCheckState(-1, TRUE); } } } PVOID CAppsListView::GetFocusedItemData() { INT item = GetSelectionMark(); if (item == -1) { return (PVOID)0; } return (PVOID)GetItemData(item); } BOOL CAppsListView::SetDisplayAppType(APPLICATION_VIEW_TYPE AppType) { ++g_AsyncIconTaskId; // Stop loading icons that are now invalid if (!DeleteAllItems()) return FALSE; ApplicationViewType = AppType; bIsAscending = TRUE; ItemCount = 0; CheckedItemCount = 0; ListView_Scroll(m_hWnd, 0, 0x7fff * -1); // FIXME: a bug in Wine ComCtl32 where VScroll is not reset after deleting items // delete old columns while (ColumnCount) { DeleteColumn(--ColumnCount); } if (!g_hDefaultPackageIcon) { ImageList_Destroy(m_hImageListView); UINT IconSize = GetSystemMetrics(SM_CXICON); UINT ilc = GetSystemColorDepth() | ILC_MASK; m_hImageListView = ImageList_Create(IconSize, IconSize, ilc, 0, 1); SetImageList(m_hImageListView, LVSIL_SMALL); SetImageList(m_hImageListView, LVSIL_NORMAL); g_hDefaultPackageIcon = (HICON)LoadImageW(hInst, MAKEINTRESOURCEW(IDI_MAIN), IMAGE_ICON, IconSize, IconSize, LR_SHARED); } ImageList_RemoveAll(m_hImageListView); g_DefaultPackageIconILIdx = ImageList_AddIcon(m_hImageListView, g_hDefaultPackageIcon); if (g_DefaultPackageIconILIdx == -1) g_DefaultPackageIconILIdx = I_IMAGENONE; // add new columns CStringW szText; switch (AppType) { case AppViewTypeInstalledApps: /* Add columns to ListView */ szText.LoadStringW(IDS_APP_NAME); AddColumn(ColumnCount++, szText, 368, LVCFMT_LEFT); szText.LoadStringW(IDS_APP_INST_VERSION); AddColumn(ColumnCount++, szText, 90, LVCFMT_RIGHT); #if 0 // This column is not currently useful for installed apps. szText.LoadStringW(IDS_APP_DESCRIPTION); AddColumn(ColumnCount++, szText, 300, LVCFMT_LEFT); #endif // Disable checkboxes ShowCheckboxes(false); break; case AppViewTypeAvailableApps: /* Add columns to ListView */ szText.LoadStringW(IDS_APP_NAME); AddColumn(ColumnCount++, szText, 250, LVCFMT_LEFT); szText.LoadStringW(IDS_APP_INST_VERSION); AddColumn(ColumnCount++, szText, 90, LVCFMT_RIGHT); szText.LoadStringW(IDS_APP_DESCRIPTION); AddColumn(ColumnCount++, szText, 300, LVCFMT_LEFT); // Enable checkboxes ShowCheckboxes(true); break; default: break; } return TRUE; } BOOL CAppsListView::SetViewMode(DWORD ViewMode) { return SendMessage(LVM_SETVIEW, (WPARAM)ViewMode, 0) == 1; } BOOL CAppsListView::AddApplication(CAppInfo *AppInfo, BOOL InitialCheckState) { if (!AppInfo) { CAsyncLoadIcon::StartTasks(); return TRUE; } int IconIndex = g_DefaultPackageIconILIdx; if (ApplicationViewType == AppViewTypeInstalledApps) { int Index = AddItem(ItemCount, IconIndex, AppInfo->szDisplayName, (LPARAM)AppInfo); if (Index == -1) return FALSE; CAsyncLoadIcon::Queue(m_hWnd, *AppInfo, true); SetItemText(Index, 1, AppInfo->szDisplayVersion.IsEmpty() ? L"---" : AppInfo->szDisplayVersion); SetItemText(Index, 2, AppInfo->szComments.IsEmpty() ? L"---" : AppInfo->szComments); ItemCount++; return TRUE; } else if (ApplicationViewType == AppViewTypeAvailableApps) { int Index = AddItem(ItemCount, IconIndex, AppInfo->szDisplayName, (LPARAM)AppInfo); if (Index == -1) return FALSE; CAsyncLoadIcon::Queue(m_hWnd, *AppInfo, false); if (InitialCheckState) { SetCheckState(Index, TRUE); } SetItemText(Index, 1, AppInfo->szDisplayVersion); SetItemText(Index, 2, AppInfo->szComments); ItemCount++; return TRUE; } else { return FALSE; } } // this function is called when parent window receiving an notification about checkstate changing VOID CAppsListView::ItemCheckStateNotify(int iItem, BOOL bCheck) { if (bCheck) { CheckedItemCount++; } else { CheckedItemCount--; } } // **** CAppsListView **** // **** CApplicationView **** BOOL CApplicationView::ProcessWindowMessage( HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam, LRESULT &theResult, DWORD dwMapId) { theResult = 0; switch (message) { case WM_CREATE: { BOOL bSuccess = TRUE; m_Panel = new CUiPanel(); m_Panel->m_VerticalAlignment = UiAlign_Stretch; m_Panel->m_HorizontalAlignment = UiAlign_Stretch; bSuccess &= CreateToolbar(); bSuccess &= CreateSearchBar(); bSuccess &= CreateComboBox(); bSuccess &= CreateHSplitter(); bSuccess &= CreateListView(); bSuccess &= CreateAppInfoDisplay(); /* APPWIZ-mode: Remove the unneeded menu items and toolbar buttons */ if (m_MainWindow->m_bAppwizMode) { HMENU hMenu; /* Delete the "Settings" item in the "File" sub-menu */ hMenu = ::GetSubMenu(m_MainWindow->GetMenu(), 0); DeleteMenuEx(hMenu, ID_SETTINGS, MF_BYCOMMAND); /* Remove the menu items: ID_INSTALL, ID_RESETDB */ hMenu = GetMenu(); DeleteMenuEx(hMenu, ID_INSTALL, MF_BYCOMMAND); DeleteMenuEx(hMenu, ID_RESETDB, MF_BYCOMMAND); /* Remove the toolbar buttons: * ID_INSTALL, ID_CHECK_ALL, ID_RESETDB * We only keep: * ID_UNINSTALL, ID_MODIFY, ID_REFRESH */ TBBUTTONINFO info = { sizeof(info), 0 }; int index; index = m_Toolbar->GetButtonInfo(ID_INSTALL, &info); if (index >= 0) m_Toolbar->DeleteButton(index); index = m_Toolbar->GetButtonInfo(ID_CHECK_ALL, &info); if (index >= 0) m_Toolbar->DeleteButton(index); index = m_Toolbar->GetButtonInfo(ID_RESETDB, &info); if (index >= 0) m_Toolbar->DeleteButton(index); /* Update the ideal width to use as a max width of buttons */ m_Toolbar->UpdateMaxButtonsWidth(); } /* Resize the toolbar */ m_Toolbar->AutoSize(); RECT rTop; ::GetWindowRect(m_Toolbar->m_hWnd, &rTop); m_HSplitter->m_Margin.top = rTop.bottom - rTop.top; if (!bSuccess) { return -1; // creation failure } } break; case WM_NOTIFY: { LPNMHDR pNotifyHeader = (LPNMHDR)lParam; if (pNotifyHeader->hwndFrom == m_ListView->GetWindow()) { switch (pNotifyHeader->code) { case LVN_ITEMCHANGED: { LPNMLISTVIEW pnic = (LPNMLISTVIEW)lParam; /* Check if this is a valid item * (technically, it can be also an unselect) */ INT ItemIndex = pnic->iItem; if (ItemIndex == -1 || ItemIndex >= ListView_GetItemCount(pnic->hdr.hwndFrom)) { break; } /* Check if the focus has been moved to another item */ if ((pnic->uChanged & LVIF_STATE) && (pnic->uNewState & LVIS_FOCUSED) && !(pnic->uOldState & LVIS_FOCUSED)) { ItemGetFocus((LPVOID)pnic->lParam); } /* Check if the item is checked/unchecked */ if (pnic->uChanged & LVIF_STATE) { int iOldState = STATEIMAGETOINDEX(pnic->uOldState); int iNewState = STATEIMAGETOINDEX(pnic->uNewState); if (iOldState == STATEIMAGE_UNCHECKED && iNewState == STATEIMAGE_CHECKED) { // this item is just checked m_ListView->ItemCheckStateNotify(pnic->iItem, TRUE); ItemCheckStateChanged(TRUE, (LPVOID)pnic->lParam); } else if (iOldState == STATEIMAGE_CHECKED && iNewState == STATEIMAGE_UNCHECKED) { // this item is just unchecked m_ListView->ItemCheckStateNotify(pnic->iItem, FALSE); ItemCheckStateChanged(FALSE, (LPVOID)pnic->lParam); } } } break; case LVN_COLUMNCLICK: { LPNMLISTVIEW pnmv = (LPNMLISTVIEW)lParam; m_ListView->ColumnClick(pnmv); } break; case NM_DBLCLK: { LPNMITEMACTIVATE Item = (LPNMITEMACTIVATE)lParam; if (Item->iItem != -1) { /* this won't do anything if the program is already installed */ if (ApplicationViewType == AppViewTypeAvailableApps) { m_MainWindow->InstallApplication( (CAppInfo *)m_ListView->GetItemData(Item->iItem)); } } } break; } } else if (pNotifyHeader->hwndFrom == m_Toolbar->GetWindow()) { switch (pNotifyHeader->code) { case TTN_GETDISPINFO: m_Toolbar->OnGetDispInfo((LPTOOLTIPTEXT)lParam); break; } } } break; case WM_SYSCOLORCHANGE: { /* Forward WM_SYSCOLORCHANGE to common controls */ m_ListView->SendMessageW(WM_SYSCOLORCHANGE, wParam, lParam); m_AppsInfo->SendMessageW(WM_SYSCOLORCHANGE, wParam, lParam); m_Toolbar->SendMessageW(WM_SYSCOLORCHANGE, wParam, lParam); m_ComboBox->SendMessageW(WM_SYSCOLORCHANGE, wParam, lParam); } break; case WM_SIZE: { OnSize(hwnd, wParam, lParam); break; } case WM_COMMAND: { OnCommand(wParam, lParam); } break; case WM_CONTEXTMENU: { bool kbd = -1 == (int)(INT_PTR)lParam; if ((HWND)wParam == m_ListView->m_hWnd) { int item = m_ListView->GetNextItem(-1, LVNI_FOCUSED | LVNI_SELECTED); if (item != -1) { POINT *ppt = NULL, pt; if (kbd) { RECT r; ListView_GetItemRect((HWND)wParam, item, &r, LVIR_LABEL); pt.x = r.left + (r.right - r.left) / 2; pt.y = r.top + (r.bottom - r.top) / 2; ::ClientToScreen((HWND)wParam, ppt = &pt); } ShowPopupMenuEx(m_hWnd, m_hWnd, 0, ID_INSTALL, ppt); return TRUE; } } } break; } return FALSE; } BOOL CApplicationView::CreateToolbar() { m_Toolbar = new CMainToolbar(); m_Toolbar->m_VerticalAlignment = UiAlign_LeftTop; m_Toolbar->m_HorizontalAlignment = UiAlign_Stretch; m_Panel->Children().Append(m_Toolbar); return m_Toolbar->Create(m_hWnd) != NULL; } BOOL CApplicationView::CreateSearchBar() { m_SearchBar = new CUiWindow(); m_SearchBar->m_VerticalAlignment = UiAlign_LeftTop; m_SearchBar->m_HorizontalAlignment = UiAlign_RightBtm; m_SearchBar->m_Margin.top = 4; m_SearchBar->m_Margin.right = TOOLBAR_PADDING; return m_SearchBar->Create(m_Toolbar->m_hWnd) != NULL; } BOOL CApplicationView::CreateComboBox() { m_ComboBox = new CUiWindow(); m_ComboBox->m_VerticalAlignment = UiAlign_LeftTop; m_ComboBox->m_HorizontalAlignment = UiAlign_RightBtm; m_ComboBox->m_Margin.top = 4; return m_ComboBox->Create(m_Toolbar->m_hWnd) != NULL; } BOOL CApplicationView::CreateHSplitter() { m_HSplitter = new CUiSplitPanel(); m_HSplitter->m_VerticalAlignment = UiAlign_Stretch; m_HSplitter->m_HorizontalAlignment = UiAlign_Stretch; m_HSplitter->m_DynamicFirst = TRUE; m_HSplitter->m_Horizontal = TRUE; m_HSplitter->m_Pos = INT_MAX; // set INT_MAX to use lowest possible position (m_MinSecond) m_HSplitter->m_MinFirst = 10; m_HSplitter->m_MinSecond = 140; m_Panel->Children().Append(m_HSplitter); return m_HSplitter->Create(m_hWnd) != NULL; } BOOL CApplicationView::CreateListView() { m_ListView = new CAppsListView(); m_ListView->m_VerticalAlignment = UiAlign_Stretch; m_ListView->m_HorizontalAlignment = UiAlign_Stretch; m_HSplitter->First().Append(m_ListView); return m_ListView->Create(m_hWnd) != NULL; } BOOL CApplicationView::CreateAppInfoDisplay() { m_AppsInfo = new CAppInfoDisplay(); m_AppsInfo->m_VerticalAlignment = UiAlign_Stretch; m_AppsInfo->m_HorizontalAlignment = UiAlign_Stretch; m_HSplitter->Second().Append(m_AppsInfo); return m_AppsInfo->Create(m_hWnd) != NULL; } void CApplicationView::SetRedraw(BOOL bRedraw) { CWindow::SetRedraw(bRedraw); m_ListView->SetRedraw(bRedraw); } void CApplicationView::SetFocusOnSearchBar() { m_SearchBar->SetFocus(); } VOID CApplicationView::OnSize(HWND hwnd, WPARAM wParam, LPARAM lParam) { if (wParam == SIZE_MINIMIZED) return; /* Resize the toolbar */ m_Toolbar->AutoSize(); /* Automatically hide captions */ DWORD dToolbarTreshold = m_Toolbar->GetMaxButtonsWidth(); DWORD dSearchbarMargin = (LOWORD(lParam) - m_SearchBar->m_Width - m_ComboBox->m_Width - TOOLBAR_PADDING * 2); if (dSearchbarMargin > dToolbarTreshold) { m_Toolbar->ShowButtonCaption(true); } else if (dSearchbarMargin < dToolbarTreshold) { m_Toolbar->ShowButtonCaption(false); } RECT r = {0, 0, LOWORD(lParam), HIWORD(lParam)}; HDWP hdwp = NULL; INT count = m_Panel->CountSizableChildren(); hdwp = BeginDeferWindowPos(count); if (hdwp) { hdwp = m_Panel->OnParentSize(r, hdwp); if (hdwp) { EndDeferWindowPos(hdwp); } } count = m_SearchBar->CountSizableChildren(); hdwp = BeginDeferWindowPos(count); if (hdwp) { hdwp = m_SearchBar->OnParentSize(r, hdwp); if (hdwp) { EndDeferWindowPos(hdwp); } } m_ComboBox->m_Margin.right = m_SearchBar->m_Width + m_SearchBar->m_Margin.right + TOOLBAR_PADDING; count = m_ComboBox->CountSizableChildren(); hdwp = BeginDeferWindowPos(count); if (hdwp) { hdwp = m_ComboBox->OnParentSize(r, hdwp); if (hdwp) { EndDeferWindowPos(hdwp); } } } VOID CApplicationView::OnCommand(WPARAM wParam, LPARAM lParam) { if (lParam) { if ((HWND)lParam == m_SearchBar->GetWindow()) { CStringW szBuf; switch (HIWORD(wParam)) { case EN_SETFOCUS: { CStringW szWndText; szBuf.LoadStringW(IDS_SEARCH_TEXT); m_SearchBar->GetWindowTextW(szWndText); if (szBuf == szWndText) { m_SearchBar->SetWindowTextW(L""); } } break; case EN_KILLFOCUS: { m_SearchBar->GetWindowTextW(szBuf); if (szBuf.IsEmpty()) { szBuf.LoadStringW(IDS_SEARCH_TEXT); m_SearchBar->SetWindowTextW(szBuf.GetString()); } } break; case EN_CHANGE: { CStringW szWndText; szBuf.LoadStringW(IDS_SEARCH_TEXT); m_SearchBar->GetWindowTextW(szWndText); if (szBuf == szWndText) { szWndText = L""; m_MainWindow->SearchTextChanged(szWndText); } else { m_MainWindow->SearchTextChanged(szWndText); } } break; } return; } else if ((HWND)lParam == m_ComboBox->GetWindow()) { int NotifyCode = HIWORD(wParam); switch (NotifyCode) { case CBN_SELCHANGE: int CurrSelection = m_ComboBox->SendMessageW(CB_GETCURSEL); int ViewModeList[] = {LV_VIEW_DETAILS, LV_VIEW_LIST, LV_VIEW_TILE}; ATLASSERT(CurrSelection < (int)_countof(ViewModeList)); if (!m_ListView->SetViewMode(ViewModeList[CurrSelection])) { MessageBoxW(L"View mode invalid or unimplemented"); } break; } return; } else if ((HWND)lParam == m_Toolbar->GetWindow()) { // the message is sent from Toolbar. fall down to continue process } else { return; } } // the LOWORD of wParam contains a Menu or Control ID WORD wCommand = LOWORD(wParam); switch (wCommand) { case ID_INSTALL: case ID_UNINSTALL: case ID_MODIFY: case ID_REGREMOVE: case ID_REFRESH: case ID_RESETDB: case ID_CHECK_ALL: m_MainWindow->SendMessageW(WM_COMMAND, wCommand, 0); break; } } CApplicationView::CApplicationView(CMainWindow *MainWindow) : m_MainWindow(MainWindow) { } CApplicationView::~CApplicationView() { delete m_Toolbar; delete m_SearchBar; delete m_ListView; delete m_AppsInfo; delete m_HSplitter; } ATL::CWndClassInfo & CApplicationView::GetWndClassInfo() { DWORD csStyle = CS_VREDRAW | CS_HREDRAW; static ATL::CWndClassInfo wc = { {sizeof(WNDCLASSEX), csStyle, StartWindowProc, 0, 0, NULL, NULL, NULL, (HBRUSH)(COLOR_BTNFACE + 1), NULL, L"RAppsApplicationView", NULL}, NULL, NULL, IDC_ARROW, TRUE, 0, _T("")}; return wc; } HWND CApplicationView::Create(HWND hwndParent) { RECT r = {0, 0, 0, 0}; // Pick the "Programs" sub-menu for building the context menu. HMENU hMenu = ::GetSubMenu(m_MainWindow->GetMenu(), 1); return CWindowImpl::Create(hwndParent, r, L"", WS_CHILD | WS_VISIBLE | WS_CLIPCHILDREN | WS_CLIPSIBLINGS, 0, hMenu); } BOOL CApplicationView::SetDisplayAppType(APPLICATION_VIEW_TYPE AppType) { if (!m_ListView->SetDisplayAppType(AppType)) { return FALSE; } ApplicationViewType = AppType; m_AppsInfo->SetWelcomeText(m_MainWindow->m_bAppwizMode); HMENU hMenu = ::GetMenu(m_hWnd); switch (AppType) { case AppViewTypeInstalledApps: { EnableMenuItem(hMenu, ID_INSTALL, MF_GRAYED); EnableMenuItem(hMenu, ID_UNINSTALL, MF_ENABLED); EnableMenuItem(hMenu, ID_MODIFY, MF_ENABLED); EnableMenuItem(hMenu, ID_REGREMOVE, MF_ENABLED); m_Toolbar->SendMessageW(TB_ENABLEBUTTON, ID_INSTALL, FALSE); m_Toolbar->SendMessageW(TB_ENABLEBUTTON, ID_UNINSTALL, TRUE); m_Toolbar->SendMessageW(TB_ENABLEBUTTON, ID_MODIFY, TRUE); m_Toolbar->SendMessageW(TB_ENABLEBUTTON, ID_CHECK_ALL, FALSE); break; } case AppViewTypeAvailableApps: { // We shouldn't get there in APPWIZ-mode. ATLASSERT(!m_MainWindow->m_bAppwizMode); EnableMenuItem(hMenu, ID_INSTALL, MF_ENABLED); EnableMenuItem(hMenu, ID_UNINSTALL, MF_GRAYED); EnableMenuItem(hMenu, ID_MODIFY, MF_GRAYED); EnableMenuItem(hMenu, ID_REGREMOVE, MF_GRAYED); m_Toolbar->SendMessageW(TB_ENABLEBUTTON, ID_INSTALL, TRUE); m_Toolbar->SendMessageW(TB_ENABLEBUTTON, ID_UNINSTALL, FALSE); m_Toolbar->SendMessageW(TB_ENABLEBUTTON, ID_MODIFY, FALSE); m_Toolbar->SendMessageW(TB_ENABLEBUTTON, ID_CHECK_ALL, TRUE); break; } } return TRUE; } BOOL CApplicationView::AddApplication(CAppInfo *AppInfo, BOOL InitialCheckState) { return m_ListView->AddApplication(AppInfo, InitialCheckState); } VOID CApplicationView::SetWatermark(const CStringW &Text) { m_ListView->SetWatermark(Text); } void CApplicationView::CheckAll() { m_ListView->CheckAll(); } PVOID CApplicationView::GetFocusedItemData() { return m_ListView->GetFocusedItemData(); } int CApplicationView::GetItemCount() { return m_ListView->GetItemCount(); } VOID CApplicationView::AppendTabOrderWindow(int Direction, ATL::CSimpleArray &TabOrderList) { m_Toolbar->AppendTabOrderWindow(Direction, TabOrderList); m_ComboBox->AppendTabOrderWindow(Direction, TabOrderList); m_SearchBar->AppendTabOrderWindow(Direction, TabOrderList); m_ListView->AppendTabOrderWindow(Direction, TabOrderList); m_AppsInfo->AppendTabOrderWindow(Direction, TabOrderList); } VOID CApplicationView::GetRestoreListSelectionData(RESTORELISTSELECTION &Restore) { LVITEMW &Item = Restore.Item; Item.mask = LVIF_TEXT|LVIF_STATE; Item.iItem = -1, Item.iSubItem = 0; Item.stateMask = LVIS_FOCUSED|LVIS_SELECTED; Item.pszText = Restore.Name, Item.cchTextMax = _countof(Restore.Name); HWND hList = m_ListView ? m_ListView->m_hWnd : NULL; if (hList) { Item.iItem = ListView_GetNextItem(hList, -1, LVNI_FOCUSED); ListView_GetItem(hList, &Item); } } VOID CApplicationView::RestoreListSelection(const RESTORELISTSELECTION &Restore) { const LVITEMW &Item = Restore.Item; int index = Item.iItem; if (index != -1) // Was there a selected item? { LVFINDINFOW fi; fi.flags = LVFI_STRING; fi.psz = Item.pszText; index = ListView_FindItem(m_ListView->m_hWnd, -1, &fi); } if (index != -1) // Is it still in the list? { ListView_SetItemState(m_ListView->m_hWnd, index, Item.state, Item.stateMask); } } // this function is called when a item of listview get focus. // CallbackParam is the param passed to listview when adding the item (the one getting focus now). VOID CApplicationView::ItemGetFocus(LPVOID CallbackParam) { if (CallbackParam) { CAppInfo *Info = static_cast(CallbackParam); m_AppsInfo->ShowAppInfo(Info); if (ApplicationViewType == AppViewTypeInstalledApps) { HMENU hMenu = ::GetMenu(m_hWnd); BOOL CanModify = Info->CanModify(); EnableMenuItem(hMenu, ID_MODIFY, CanModify ? MF_ENABLED : MF_GRAYED); m_Toolbar->SendMessageW(TB_ENABLEBUTTON, ID_MODIFY, CanModify); } } } // this function is called when a item of listview is checked/unchecked // CallbackParam is the param passed to listview when adding the item (the one getting changed now). VOID CApplicationView::ItemCheckStateChanged(BOOL bChecked, LPVOID CallbackParam) { m_MainWindow->ItemCheckStateChanged(bChecked, CallbackParam); } // **** CApplicationView ****