/* Copyright (C) 2000 Kimmo Pekkola This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ /* $Header: /home/cvsroot/Rainmeter/Library/AboutDialog.cpp,v 1.1.1.1 2005/07/10 18:51:06 rainy Exp $ $Log: AboutDialog.cpp,v $ Revision 1.1.1.1 2005/07/10 18:51:06 rainy no message Revision 1.6 2004/08/13 15:43:19 rainy Added update check. Revision 1.5 2004/06/05 10:55:54 rainy Too much changes to be listed in here... Revision 1.4 2004/03/13 16:13:50 rainy Added support for multiple configs. Revision 1.3 2003/12/05 15:50:09 Rainy Multi-instance changes. Revision 1.2 2002/12/23 14:26:33 rainy Made the dialog resizable. Revision 1.1 2002/07/01 15:35:55 rainy Intial version */ #pragma warning(disable: 4786) #pragma warning(disable: 4996) #include "Rainmeter.h" #include "MeterWindow.h" #include "Measure.h" #include "resource.h" #include "AboutDialog.h" #include extern CRainmeter* Rainmeter; INT_PTR CALLBACK AboutProc(HWND hwndDlg, UINT message, WPARAM wParam, LPARAM lParam); HWND g_DialogWin = NULL; struct PLUGIN_INFO { std::wstring name; UINT version; std::wstring author; }; std::vector g_Plugins; HWND OpenAboutDialog(HWND hwndOwner, HINSTANCE instance) { if (g_DialogWin == NULL) { g_DialogWin = CreateDialog(instance, MAKEINTRESOURCE(IDD_ABOUT_DIALOG), hwndOwner, AboutProc); } ShowWindow(g_DialogWin, SW_SHOWNORMAL); UpdateAboutStatistics(); return g_DialogWin; } void UpdateAboutStatistics() { if (g_DialogWin != NULL && IsWindowVisible(g_DialogWin)) { HWND widget; widget = GetDlgItem(g_DialogWin, IDC_CONFIG_TAB); int selected = TabCtrl_GetCurSel(widget); int current = 0; widget = GetDlgItem(g_DialogWin, IDC_STATISTICS); SendMessage(widget, WM_SETREDRAW, 0, 0); std::map& windows = Rainmeter->GetAllMeterWindows(); std::map::iterator iter = windows.begin(); for( ; iter != windows.end(); iter++) { if (current == selected) { int count = ListView_GetItemCount(widget); CMeterWindow* meterWindow = (*iter).second; std::list& measures = meterWindow->GetMeasures(); int index = 0; std::list::iterator i = measures.begin(); for( ; i != measures.end(); i++) { const WCHAR* name = (*i)->GetName(); const WCHAR* val = (*i)->GetStats(); std::wstring range; WCHAR buffer[256]; double minVal = (*i)->GetMinValue(); double maxVal = (*i)->GetMaxValue(); CMeasure::GetScaledValue(1, minVal, buffer); range = buffer; range += L" - "; CMeasure::GetScaledValue(1, maxVal, buffer); range += buffer; if (name && wcslen(name) > 0) { if (index < count) { ListView_SetItemText(widget, index, 0, (WCHAR*)name); } else { LVITEM vitem; vitem.mask = LVIF_TEXT; vitem.iItem = 0; vitem.iSubItem = 0; vitem.pszText = (WCHAR*)name; ListView_InsertItem(widget, &vitem); } if (val && wcslen(val) > 0) { ListView_SetItemText(widget, index, 1, (WCHAR*)val); } ListView_SetItemText(widget, index, 2, (WCHAR*)range.c_str()); index++; } } break; } current++; } SendMessage(widget, WM_SETREDRAW, 1, 0); } } void UpdateWidgets(HWND window) { HWND widget; widget = GetDlgItem(g_DialogWin, IDC_CONFIG_TAB); int selected = TabCtrl_GetCurSel(widget); int count = TabCtrl_GetItemCount(widget); widget = GetDlgItem(g_DialogWin, IDC_STATISTICS); ListView_DeleteAllItems(widget); if (count == selected + 1) { LVCOLUMN lvc; lvc.mask = LVCF_TEXT; lvc.pszText = L"Plugin"; ListView_SetColumn(widget, 0, &lvc); lvc.pszText = L"Version"; ListView_SetColumn(widget, 1, &lvc); lvc.pszText = L"Author"; ListView_SetColumn(widget, 2, &lvc); // Update the list of plugins std::vector::iterator iter = g_Plugins.begin(); LVITEM vitem; vitem.mask = LVIF_TEXT; int i = 0; for ( ; iter != g_Plugins.end(); iter++) { if (!(*iter).name.empty()) { vitem.iItem = i; vitem.iSubItem = 0; vitem.pszText = (WCHAR*)(*iter).name.c_str(); ListView_InsertItem(widget, &vitem); } if ((*iter).version != 0) { WCHAR buffer[256]; swprintf(buffer, L"%i.%i", (*iter).version / 1000, (*iter).version % 1000); ListView_SetItemText(widget, i, 1, buffer); } ListView_SetItemText(widget, i, 2, (WCHAR*)(*iter).author.c_str()); i++; } if (g_Plugins.size() > 0) { ListView_SetItemState(widget, 0, LVIS_SELECTED, LVIS_SELECTED); } } else { LVCOLUMN lvc; lvc.mask = LVCF_TEXT; lvc.pszText = L"Measure"; ListView_SetColumn(widget, 0, &lvc); lvc.pszText = L"Value"; ListView_SetColumn(widget, 1, &lvc); lvc.pszText = L"Range"; ListView_SetColumn(widget, 2, &lvc); } } typedef LPCTSTR (*GETPLUGINAUTHOR)(); typedef UINT (*GETPLUGINVERSION)(); void ScanPlugins() { WIN32_FIND_DATA fileData; // Data structure describes the file found HANDLE hSearch; // Search handle returned by FindFirstFile std::wstring files = CRainmeter::FixPath(L"*.dll", PATH_FOLDER_PLUGIN, L""); g_Plugins.clear(); // Start searching for .ini files in the given directory. hSearch = FindFirstFile(files.c_str(), &fileData); do { if(hSearch == INVALID_HANDLE_VALUE) break; // No more files found PLUGIN_INFO info; info.name = fileData.cFileName; info.version = 0; // Try to get the version and author std::wstring tmpSz = CRainmeter::FixPath(fileData.cFileName, PATH_FOLDER_PLUGIN, L""); HMODULE dll = LoadLibrary(tmpSz.c_str()); if (dll) { GETPLUGINAUTHOR GetAuthorFunc = (GETPLUGINAUTHOR)GetProcAddress(dll, "GetPluginAuthor"); if (GetAuthorFunc) { LPCTSTR author = GetAuthorFunc(); if (author && wcslen(author) > 0) { info.author = author; } } GETPLUGINVERSION GetVersionFunc = (GETPLUGINVERSION)GetProcAddress(dll, "GetPluginVersion"); if (GetVersionFunc) { info.version = GetVersionFunc(); } FreeLibrary(dll); } g_Plugins.push_back(info); } while(FindNextFile(hSearch, &fileData)); FindClose(hSearch); } void RepositionControls(HWND hwndDlg) { RECT r, br, ar, sr, wr; HWND widget; GetClientRect(hwndDlg, &r); GetClientRect(GetDlgItem(hwndDlg, IDOK), &br); GetClientRect(GetDlgItem(hwndDlg, IDC_STATIC_ABOUT), &ar); GetClientRect(GetDlgItem(hwndDlg, IDC_VERSION_STRING), &sr); // Reposition the statistics widgets widget = GetDlgItem(hwndDlg, IDC_STATIC_ABOUT); SetWindowPos(widget, NULL, 0, 0, r.right - 22, ar.bottom, SWP_NOMOVE | SWP_NOZORDER); widget = GetDlgItem(hwndDlg, IDC_VERSION_STRING); SetWindowPos(widget, NULL, 0, 0, r.right - 44, sr.bottom, SWP_NOMOVE | SWP_NOZORDER); widget = GetDlgItem(hwndDlg, IDC_BUILD_STRING); SetWindowPos(widget, NULL, 0, 0, r.right - 44, sr.bottom, SWP_NOMOVE | SWP_NOZORDER); widget = GetDlgItem(hwndDlg, IDC_URL_STRING); SetWindowPos(widget, NULL, 0, 0, r.right - 44, sr.bottom, SWP_NOMOVE | SWP_NOZORDER); widget = GetDlgItem(hwndDlg, IDC_CHECK_FOR_UPDATE); GetClientRect(widget, &wr); MapWindowPoints(widget, hwndDlg, (LPPOINT)&wr, 2); SetWindowPos(widget, NULL, (r.right - (wr.right - wr.left)) / 2, wr.top, 0, 0, SWP_NOSIZE | SWP_NOZORDER); widget = GetDlgItem(hwndDlg, IDC_CONFIG_TAB); SetWindowPos(widget, NULL, 0, 0, r.right - 22, r.bottom - 175, SWP_NOMOVE | SWP_NOZORDER); widget = GetDlgItem(hwndDlg, IDC_STATISTICS); SetWindowPos(widget, NULL, 0, 0, r.right - 44, r.bottom - 210, SWP_NOMOVE | SWP_NOZORDER); widget = GetDlgItem(hwndDlg, IDOK); SetWindowPos(widget, NULL, (r.right - br.right) / 2, r.bottom - br.bottom - 11, br.right, br.bottom, SWP_NOZORDER); } BOOL OnInitAboutDialog(HWND window) { WCHAR tmpSz[256]; HWND widget; widget = GetDlgItem(window, IDC_VERSION_STRING); swprintf(tmpSz, L"%s version %s", APPNAME, APPVERSION); SetWindowText(widget, tmpSz); widget = GetDlgItem(window, IDC_BUILD_STRING); swprintf(tmpSz, L"Build on %s", ConvertToWide(__DATE__).c_str()); SetWindowText(widget, tmpSz); // Add tabs for each config widget = GetDlgItem(window, IDC_CONFIG_TAB); TCITEM tie; tie.mask = TCIF_TEXT; std::map& windows = Rainmeter->GetAllMeterWindows(); std::map::iterator iter = windows.begin(); int i = 0; for( ; iter != windows.end(); iter++) { CMeterWindow* meterWindow = (*iter).second; tie.pszText = (WCHAR*)meterWindow->GetSkinName().c_str(); TabCtrl_InsertItem(widget, i++, &tie); } tie.pszText = L"Plugins"; TabCtrl_InsertItem(widget, i, &tie); // Add columns to the list view widget = GetDlgItem(window, IDC_STATISTICS); ListView_SetExtendedListViewStyleEx(widget, LVS_EX_FULLROWSELECT, LVS_EX_FULLROWSELECT); LVCOLUMN lvc; lvc.mask = LVCF_FMT | LVCF_WIDTH | LVCF_TEXT | LVCF_SUBITEM; lvc.iSubItem = 0; lvc.pszText = L"Measure"; lvc.cx = 150; lvc.fmt = LVCFMT_LEFT; // left-aligned column ListView_InsertColumn(widget, 0, &lvc); lvc.iSubItem = 1; lvc.cx = 130; lvc.pszText = L"Value"; ListView_InsertColumn(widget, 1, &lvc); lvc.iSubItem = 1; lvc.cx = 130; lvc.pszText = L"Range"; ListView_InsertColumn(widget, 2, &lvc); CheckDlgButton(window, IDC_CHECK_FOR_UPDATE, Rainmeter->GetCheckUpdate() ? BST_CHECKED : BST_UNCHECKED); ScanPlugins(); UpdateWidgets(window); RepositionControls(window); g_DialogWin = window; return TRUE; } INT_PTR CALLBACK AboutProc(HWND hwndDlg, UINT message, WPARAM wParam, LPARAM lParam) { switch (message) { case WM_INITDIALOG: return OnInitAboutDialog(hwndDlg); case WM_WINDOWPOSCHANGING: { WINDOWPOS* pos = (WINDOWPOS*)lParam; pos->cx = max(280, pos->cx); pos->cy = max(280, pos->cy); } break; case WM_SIZE: RepositionControls(hwndDlg); break; case WM_NOTIFY: { LPNMHDR lpnmhdr = (LPNMHDR)lParam; if (lpnmhdr->code == TCN_SELCHANGE) { UpdateWidgets(hwndDlg); UpdateAboutStatistics(); } } break; case WM_CLOSE: Rainmeter->SaveSettings(); DestroyWindow(hwndDlg); g_DialogWin = NULL; return TRUE; case WM_COMMAND: switch (LOWORD(wParam)) { case IDC_CHECK_FOR_UPDATE: if (IsDlgButtonChecked(hwndDlg, IDC_CHECK_FOR_UPDATE)) { Rainmeter->SetCheckUpdate(TRUE); } else { Rainmeter->SetCheckUpdate(FALSE); } break; case IDOK: Rainmeter->SaveSettings(); DestroyWindow(hwndDlg); g_DialogWin = NULL; return TRUE; } break; } return FALSE; }