2010-09-23 08:49:43 +00:00
|
|
|
/*
|
2012-01-23 06:36:15 +00:00
|
|
|
Copyright (C) 2005 Kimmo Pekkola, 2009 Greg Schoppe
|
2010-09-23 08:49:43 +00:00
|
|
|
|
2012-01-23 06:36:15 +00:00
|
|
|
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.
|
2010-09-23 08:49:43 +00:00
|
|
|
|
2012-01-23 06:36:15 +00:00
|
|
|
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.
|
2010-09-23 08:49:43 +00:00
|
|
|
|
2012-01-23 06:36:15 +00:00
|
|
|
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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2010-09-23 08:49:43 +00:00
|
|
|
*/
|
|
|
|
|
2012-02-13 16:45:11 +00:00
|
|
|
#define WIN32_LEAN_AND_MEAN
|
|
|
|
#include <Windows.h>
|
|
|
|
#include <ShellAPI.h>
|
|
|
|
#include <ShlObj.h>
|
|
|
|
#include <process.h>
|
|
|
|
#include <vector>
|
|
|
|
#include "../../Library/Export.h" // Rainmeter's exported functions
|
2011-02-03 18:09:24 +00:00
|
|
|
#include "../../Library/DisableThreadLibraryCalls.h" // contains DllMain entry point
|
2011-01-30 09:31:41 +00:00
|
|
|
|
2012-02-13 16:45:11 +00:00
|
|
|
struct MeasureData
|
2010-09-23 08:49:43 +00:00
|
|
|
{
|
2012-02-13 16:45:11 +00:00
|
|
|
bool count;
|
|
|
|
|
|
|
|
MeasureData() : count(false) {}
|
2010-09-23 08:49:43 +00:00
|
|
|
};
|
|
|
|
|
2012-02-13 16:45:11 +00:00
|
|
|
struct BinData
|
2010-09-23 08:49:43 +00:00
|
|
|
{
|
2012-02-13 16:45:11 +00:00
|
|
|
FILETIME lastAccess;
|
|
|
|
HANDLE directory;
|
|
|
|
WCHAR drive;
|
2012-01-08 17:35:29 +00:00
|
|
|
};
|
2011-03-29 19:21:57 +00:00
|
|
|
|
2012-02-13 16:45:11 +00:00
|
|
|
unsigned int __stdcall QueryRecycleBinThreadProc(void* pParam);
|
|
|
|
HRESULT GetFolderCLSID(LPCWSTR pszPath, CLSID* pathCLSID);
|
|
|
|
HANDLE GetRecycleBinHandle(WCHAR drive);
|
|
|
|
|
|
|
|
std::vector<BinData> g_BinData;
|
|
|
|
double g_BinCount = 0;
|
|
|
|
double g_BinSize = 0;
|
|
|
|
|
|
|
|
UINT g_InstanceCount = 0;
|
|
|
|
HANDLE g_Thread = NULL;
|
|
|
|
|
2012-02-02 07:39:14 +00:00
|
|
|
PLUGIN_EXPORT void Initialize(void** data, void* rm)
|
2012-01-08 17:35:29 +00:00
|
|
|
{
|
|
|
|
MeasureData* measure = new MeasureData;
|
|
|
|
*data = measure;
|
2012-02-13 16:45:11 +00:00
|
|
|
|
|
|
|
++g_InstanceCount;
|
2010-09-23 08:49:43 +00:00
|
|
|
}
|
|
|
|
|
2012-01-08 17:35:29 +00:00
|
|
|
PLUGIN_EXPORT void Reload(void* data, void* rm, double* maxValue)
|
2010-09-23 08:49:43 +00:00
|
|
|
{
|
2012-01-08 17:35:29 +00:00
|
|
|
MeasureData* measure = (MeasureData*)data;
|
2011-03-29 19:21:57 +00:00
|
|
|
|
2012-01-08 17:35:29 +00:00
|
|
|
LPCWSTR value = RmReadString(rm, L"RecycleType", L"COUNT");
|
|
|
|
if (_wcsicmp(L"COUNT", value) == 0)
|
|
|
|
{
|
2012-02-13 16:45:11 +00:00
|
|
|
measure->count = true;
|
2012-01-08 17:35:29 +00:00
|
|
|
}
|
|
|
|
else if (_wcsicmp(L"SIZE", value) == 0)
|
2010-09-23 08:49:43 +00:00
|
|
|
{
|
2012-02-13 16:45:11 +00:00
|
|
|
measure->count = false;
|
2010-09-23 08:49:43 +00:00
|
|
|
}
|
2012-01-08 17:35:29 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
WCHAR buffer[256];
|
|
|
|
_snwprintf_s(buffer, _TRUNCATE, L"RecycleManager.dll: RecycleType=%s is not valid in [%s]", value, RmGetMeasureName(rm));
|
|
|
|
RmLog(LOG_ERROR, buffer);
|
|
|
|
}
|
2010-09-23 08:49:43 +00:00
|
|
|
}
|
|
|
|
|
2012-01-08 17:35:29 +00:00
|
|
|
PLUGIN_EXPORT double Update(void* data)
|
2010-09-23 08:49:43 +00:00
|
|
|
{
|
2012-01-08 17:35:29 +00:00
|
|
|
MeasureData* measure = (MeasureData*)data;
|
2011-03-29 19:21:57 +00:00
|
|
|
|
2012-02-13 16:45:11 +00:00
|
|
|
if (!g_Thread)
|
2010-09-23 08:49:43 +00:00
|
|
|
{
|
2012-02-13 16:45:11 +00:00
|
|
|
WCHAR buffer[128];
|
|
|
|
DWORD len = GetLogicalDriveStrings(128, buffer);
|
|
|
|
|
|
|
|
// Convert "A:\\\0B:\\\0" -> "AB\0"
|
|
|
|
int index = 0;
|
|
|
|
for (int i = 0; i < len; i += 4)
|
2010-09-23 08:49:43 +00:00
|
|
|
{
|
2012-02-13 16:45:11 +00:00
|
|
|
buffer[index] = buffer[i];
|
|
|
|
++index;
|
2010-09-23 08:49:43 +00:00
|
|
|
}
|
2012-02-13 16:45:11 +00:00
|
|
|
buffer[index] = L'\0';
|
|
|
|
|
|
|
|
const WCHAR DRIVE_HANDLED = 1;
|
|
|
|
bool changed = false;
|
|
|
|
auto iter = g_BinData.begin();
|
|
|
|
while (iter != g_BinData.end())
|
2010-09-23 08:49:43 +00:00
|
|
|
{
|
2012-02-13 16:45:11 +00:00
|
|
|
BinData& data = (*iter);
|
|
|
|
|
|
|
|
WCHAR* pos = wcschr(buffer, data.drive);
|
|
|
|
if (pos != NULL)
|
2011-10-12 13:34:04 +00:00
|
|
|
{
|
2012-02-13 16:45:11 +00:00
|
|
|
*pos = DRIVE_HANDLED;
|
|
|
|
++iter;
|
|
|
|
|
|
|
|
if (data.lastAccess.dwHighDateTime != 0xFFFFFFFF &&
|
|
|
|
data.lastAccess.dwLowDateTime != 0xFFFFFFFF)
|
2011-10-12 13:34:04 +00:00
|
|
|
{
|
2012-02-13 16:45:11 +00:00
|
|
|
FILETIME lastAccess;
|
|
|
|
GetFileTime(data.directory, NULL, &lastAccess, NULL);
|
|
|
|
if (data.lastAccess.dwHighDateTime != lastAccess.dwHighDateTime ||
|
|
|
|
data.lastAccess.dwLowDateTime != lastAccess.dwLowDateTime)
|
|
|
|
{
|
|
|
|
data.lastAccess.dwHighDateTime = lastAccess.dwHighDateTime;
|
|
|
|
data.lastAccess.dwLowDateTime = lastAccess.dwLowDateTime;
|
|
|
|
changed = true;
|
|
|
|
}
|
2011-10-12 13:34:04 +00:00
|
|
|
}
|
2012-02-13 16:45:11 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Drive removed
|
|
|
|
if (data.directory)
|
2011-10-12 13:34:04 +00:00
|
|
|
{
|
2012-02-13 16:45:11 +00:00
|
|
|
CloseHandle(data.directory);
|
2011-10-12 13:34:04 +00:00
|
|
|
}
|
2012-02-13 16:45:11 +00:00
|
|
|
|
|
|
|
iter = g_BinData.erase(iter);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < index; ++i)
|
|
|
|
{
|
|
|
|
if (buffer[i] != DRIVE_HANDLED)
|
|
|
|
{
|
|
|
|
// New drive
|
|
|
|
BinData data = {0};
|
|
|
|
data.directory = GetRecycleBinHandle(buffer[i]);
|
|
|
|
data.drive = buffer[i];
|
|
|
|
data.lastAccess.dwHighDateTime =
|
|
|
|
data.lastAccess.dwLowDateTime = data.directory ? 0 : 0xFFFFFFFF;
|
|
|
|
|
|
|
|
g_BinData.push_back(data);
|
2011-10-12 13:34:04 +00:00
|
|
|
}
|
2012-02-13 16:45:11 +00:00
|
|
|
}
|
2012-01-08 17:35:29 +00:00
|
|
|
|
2012-02-13 16:45:11 +00:00
|
|
|
if (changed)
|
|
|
|
{
|
|
|
|
g_Thread = (HANDLE)_beginthreadex(NULL, 64 * 1024, QueryRecycleBinThreadProc, NULL, 0, NULL);
|
2010-09-23 08:49:43 +00:00
|
|
|
}
|
2012-02-13 16:45:11 +00:00
|
|
|
|
2010-09-23 08:49:43 +00:00
|
|
|
}
|
2011-03-29 19:21:57 +00:00
|
|
|
|
2012-02-13 16:45:11 +00:00
|
|
|
return measure->count ? g_BinCount : g_BinSize;
|
2010-09-23 08:49:43 +00:00
|
|
|
}
|
|
|
|
|
2012-01-08 17:35:29 +00:00
|
|
|
PLUGIN_EXPORT void Finalize(void* data)
|
2010-09-23 08:49:43 +00:00
|
|
|
{
|
2012-01-08 17:35:29 +00:00
|
|
|
MeasureData* measure = (MeasureData*)data;
|
|
|
|
delete measure;
|
2010-09-23 08:49:43 +00:00
|
|
|
|
2012-02-13 16:45:11 +00:00
|
|
|
--g_InstanceCount;
|
|
|
|
if (g_InstanceCount == 0)
|
2011-10-12 13:34:04 +00:00
|
|
|
{
|
2012-02-13 16:45:11 +00:00
|
|
|
for (auto iter = g_BinData.cbegin(); iter != g_BinData.cend(); ++iter)
|
2010-09-23 08:49:43 +00:00
|
|
|
{
|
2012-02-13 16:45:11 +00:00
|
|
|
if ((*iter).directory)
|
2011-10-12 13:34:04 +00:00
|
|
|
{
|
2012-02-13 16:45:11 +00:00
|
|
|
CloseHandle((*iter).directory);
|
2011-10-12 13:34:04 +00:00
|
|
|
}
|
2010-09-23 08:49:43 +00:00
|
|
|
}
|
2012-02-13 16:45:11 +00:00
|
|
|
|
|
|
|
WaitForSingleObject(g_Thread, INFINITE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
PLUGIN_EXPORT void ExecuteBang(void* data, LPCWSTR args)
|
|
|
|
{
|
|
|
|
MeasureData* measure = (MeasureData*)data;
|
|
|
|
|
2012-01-08 17:35:29 +00:00
|
|
|
if (_wcsicmp(args, L"EmptyBin") == 0)
|
|
|
|
{
|
2012-02-13 16:45:11 +00:00
|
|
|
SHEmptyRecycleBin(NULL, NULL, 0);
|
2011-03-29 19:21:57 +00:00
|
|
|
}
|
2012-01-08 17:35:29 +00:00
|
|
|
else if (_wcsicmp(args, L"EmptyBinSilent") == 0)
|
2011-03-29 19:21:57 +00:00
|
|
|
{
|
2012-02-13 16:45:11 +00:00
|
|
|
SHEmptyRecycleBin(NULL, NULL, SHERB_NOCONFIRMATION | SHERB_NOPROGRESSUI | SHERB_NOSOUND);
|
2012-01-08 17:35:29 +00:00
|
|
|
}
|
|
|
|
else if (_wcsicmp(args, L"OpenBin") == 0)
|
|
|
|
{
|
|
|
|
ShellExecute(NULL, L"open", L"explorer.exe", L"/N,::{645FF040-5081-101B-9F08-00AA002F954E}", NULL, SW_SHOW);
|
2011-03-29 19:21:57 +00:00
|
|
|
}
|
2010-09-23 08:49:43 +00:00
|
|
|
}
|
2012-02-13 16:45:11 +00:00
|
|
|
|
|
|
|
unsigned int __stdcall QueryRecycleBinThreadProc(void* pParam)
|
|
|
|
{
|
|
|
|
SHQUERYRBINFO rbi = {0};
|
|
|
|
rbi.cbSize = sizeof(SHQUERYRBINFO);
|
|
|
|
SHQueryRecycleBin(NULL, &rbi);
|
|
|
|
g_BinCount = (double)rbi.i64NumItems;
|
|
|
|
g_BinSize = (double)rbi.i64Size;
|
|
|
|
|
|
|
|
g_Thread = NULL;
|
|
|
|
|
|
|
|
_endthreadex(0);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT GetFolderCLSID(LPCWSTR path, CLSID* clsid)
|
|
|
|
{
|
|
|
|
LPITEMIDLIST pidl;
|
|
|
|
HRESULT hr = SHParseDisplayName(path, NULL, &pidl, 0, NULL);
|
|
|
|
if (SUCCEEDED(hr))
|
|
|
|
{
|
|
|
|
IShellFolder* sf;
|
|
|
|
LPCITEMIDLIST pidlLast;
|
|
|
|
hr = SHBindToParent(pidl, IID_IShellFolder, (void**)&sf, &pidlLast);
|
|
|
|
if (SUCCEEDED(hr))
|
|
|
|
{
|
|
|
|
SHDESCRIPTIONID did;
|
|
|
|
hr = SHGetDataFromIDList(sf, pidlLast, SHGDFIL_DESCRIPTIONID, &did, sizeof(did));
|
|
|
|
*clsid = did.clsid;
|
|
|
|
|
|
|
|
sf->Release();
|
|
|
|
}
|
|
|
|
|
|
|
|
CoTaskMemFree(pidl);
|
|
|
|
}
|
|
|
|
return hr;
|
|
|
|
}
|
|
|
|
|
|
|
|
HANDLE GetRecycleBinHandle(WCHAR drive)
|
|
|
|
{
|
|
|
|
WCHAR search[] = L"\0:\\*";
|
|
|
|
search[0] = drive;
|
|
|
|
|
|
|
|
WIN32_FIND_DATA fd;
|
|
|
|
HANDLE hSearch = FindFirstFile(search, &fd);
|
|
|
|
if (hSearch != INVALID_HANDLE_VALUE)
|
|
|
|
{
|
|
|
|
bool found = false;
|
|
|
|
std::wstring path;
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
if ((fd.dwFileAttributes &
|
|
|
|
(FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN | FILE_ATTRIBUTE_DIRECTORY)) == 0)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
WCHAR buffer[MAX_PATH];
|
|
|
|
int len = _snwprintf_s(buffer, _TRUNCATE, L"%c:\\%s\\*", drive, fd.cFileName);
|
|
|
|
|
|
|
|
// Find the SID-specific child directory
|
|
|
|
HANDLE hChildSearch = FindFirstFile(buffer, &fd);
|
|
|
|
if (hChildSearch != INVALID_HANDLE_VALUE)
|
|
|
|
{
|
|
|
|
do
|
|
|
|
{
|
|
|
|
if (!(fd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) || (fd.cFileName[0] == L'.'))
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
path.assign(buffer, len - 1);
|
|
|
|
path += fd.cFileName;
|
|
|
|
|
|
|
|
CLSID id;
|
|
|
|
HRESULT hr = GetFolderCLSID(path.c_str(), &id);
|
|
|
|
if (SUCCEEDED(hr) && IsEqualGUID(CLSID_RecycleBin, id))
|
|
|
|
{
|
|
|
|
found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
while (FindNextFile(hChildSearch, &fd));
|
|
|
|
|
|
|
|
FindClose(hChildSearch);
|
|
|
|
|
|
|
|
if (found)
|
|
|
|
{
|
|
|
|
// Break out of main loop
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
while (FindNextFile(hSearch, &fd));
|
|
|
|
|
|
|
|
FindClose(hSearch);
|
|
|
|
|
|
|
|
if (found)
|
|
|
|
{
|
|
|
|
HANDLE hDir = CreateFile(
|
|
|
|
path.c_str(),
|
|
|
|
GENERIC_READ,
|
|
|
|
FILE_SHARE_READ | FILE_SHARE_WRITE,
|
|
|
|
NULL,
|
|
|
|
OPEN_EXISTING,
|
|
|
|
FILE_FLAG_BACKUP_SEMANTICS,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
return hDir;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|