mirror of
https://github.com/chibicitiberiu/rainmeter-studio.git
synced 2024-02-24 04:33:31 +00:00
FileView: Added Recursive=2 option. Commands FollowPath and PreviousFolder do not work with this option. Also, ShowFile, ShowFolder, and ShowDotDot have no effect on this option.
Drives are now sorted. Folders have dates and times. Lots of tweaks and cleanup. Updated to version 2.
This commit is contained in:
parent
53f5ceb674
commit
dee19de15c
@ -44,9 +44,9 @@ typedef struct // 22 bytes
|
|||||||
#pragma pack(pop)
|
#pragma pack(pop)
|
||||||
|
|
||||||
unsigned __stdcall SystemThreadProc(void* pParam);
|
unsigned __stdcall SystemThreadProc(void* pParam);
|
||||||
void GetSubFolderSize(const std::vector<FileInfo> files, const std::wstring path, int& folderCount, int& fileCount, UINT64& folderSize);
|
void GetFolderInfo(std::queue<std::wstring>& folderQueue, std::wstring& folder, ParentMeasure* parent, RecursiveType rType);
|
||||||
void GetIcon(std::wstring fileName, std::wstring iconPath, IconSize iconSize);
|
void GetIcon(std::wstring filePath, std::wstring iconPath, IconSize iconSize);
|
||||||
HRESULT SaveIcon(HICON icon, FILE*fp);
|
HRESULT SaveIcon(HICON hIcon, FILE* fp);
|
||||||
|
|
||||||
std::vector<ParentMeasure*> g_ParentMeasures;
|
std::vector<ParentMeasure*> g_ParentMeasures;
|
||||||
static CRITICAL_SECTION g_CriticalSection;
|
static CRITICAL_SECTION g_CriticalSection;
|
||||||
@ -87,12 +87,12 @@ PLUGIN_EXPORT void Reload(void* data, void* rm, double* maxValue)
|
|||||||
{
|
{
|
||||||
path = path.substr(1, path.size() - 2);
|
path = path.substr(1, path.size() - 2);
|
||||||
|
|
||||||
for (auto iter = g_ParentMeasures.begin(); iter != g_ParentMeasures.end(); ++iter)
|
for (auto iter : g_ParentMeasures)
|
||||||
{
|
{
|
||||||
if (_wcsicmp((*iter)->name, path.c_str()) == 0 &&
|
if (_wcsicmp(iter->name, path.c_str()) == 0 &&
|
||||||
(*iter)->skin == skin)
|
iter->skin == skin)
|
||||||
{
|
{
|
||||||
child->parent = (*iter);
|
child->parent = iter;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -157,14 +157,32 @@ PLUGIN_EXPORT void Reload(void* data, void* rm, double* maxValue)
|
|||||||
int count = RmReadInt(rm, L"Count", 1);
|
int count = RmReadInt(rm, L"Count", 1);
|
||||||
child->parent->count = count > 0 ? count : 1;
|
child->parent->count = count > 0 ? count : 1;
|
||||||
|
|
||||||
child->parent->isRecursive = 0!=RmReadInt(rm, L"Recursive", 0);
|
int recursive = RmReadInt(rm, L"Recursive", 0);
|
||||||
|
switch (recursive)
|
||||||
|
{
|
||||||
|
default:
|
||||||
|
RmLog(LOG_WARNING, L"Invalid Recursive type");
|
||||||
|
|
||||||
|
case 0:
|
||||||
|
child->parent->recursiveType = RECURSIVE_NONE;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 1:
|
||||||
|
child->parent->recursiveType = RECURSIVE_PARTIAL;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 2:
|
||||||
|
child->parent->recursiveType = RECURSIVE_FULL;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
child->parent->sortAscending = 0!=RmReadInt(rm, L"SortAscending", 1);
|
child->parent->sortAscending = 0!=RmReadInt(rm, L"SortAscending", 1);
|
||||||
child->parent->showDotDot = 0!=RmReadInt(rm, L"ShowDotDot", 1);
|
child->parent->showDotDot = 0!=RmReadInt(rm, L"ShowDotDot", 1);
|
||||||
child->parent->showFile = 0!=RmReadInt(rm, L"ShowFile", 1);
|
child->parent->showFile = 0!=RmReadInt(rm, L"ShowFile", 1);
|
||||||
child->parent->showFolder = 0!=RmReadInt(rm, L"ShowFolder", 1);
|
child->parent->showFolder = 0!=RmReadInt(rm, L"ShowFolder", 1);
|
||||||
child->parent->showHidden = 0!=RmReadInt(rm, L"ShowHidden", 1);
|
child->parent->showHidden = 0!=RmReadInt(rm, L"ShowHidden", 1);
|
||||||
child->parent->showSystem = 0!=RmReadInt(rm, L"ShowSystem", 0);
|
child->parent->showSystem = 0!=RmReadInt(rm, L"ShowSystem", 0);
|
||||||
child->parent->hideExtension = 0!=RmReadInt(rm, L"HideExtensions", 0);
|
child->parent->hideExtension = 0!=RmReadInt(rm, L"HideExtensions", 0);
|
||||||
child->parent->extensions = Tokenize(RmReadString(rm, L"Extensions", L""), L";");
|
child->parent->extensions = Tokenize(RmReadString(rm, L"Extensions", L""), L";");
|
||||||
|
|
||||||
child->parent->wildcardSearch = RmReadString(rm, L"WildcardSearch", L"*");
|
child->parent->wildcardSearch = RmReadString(rm, L"WildcardSearch", L"*");
|
||||||
@ -172,12 +190,6 @@ PLUGIN_EXPORT void Reload(void* data, void* rm, double* maxValue)
|
|||||||
child->parent->finishAction = RmReadString(rm, L"FinishAction", L"", false);
|
child->parent->finishAction = RmReadString(rm, L"FinishAction", L"", false);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto iter = std::find(child->parent->children.begin(), child->parent->children.end(), child);
|
|
||||||
if (iter == child->parent->children.end())
|
|
||||||
{
|
|
||||||
child->parent->children.push_back(child);
|
|
||||||
}
|
|
||||||
|
|
||||||
int index = RmReadInt(rm, L"Index", 1) - 1;
|
int index = RmReadInt(rm, L"Index", 1) - 1;
|
||||||
child->index = index >= 0 ? index : 1;
|
child->index = index >= 0 ? index : 1;
|
||||||
|
|
||||||
@ -238,7 +250,7 @@ PLUGIN_EXPORT void Reload(void* data, void* rm, double* maxValue)
|
|||||||
WCHAR buffer[MAX_PATH];
|
WCHAR buffer[MAX_PATH];
|
||||||
_itow_s(child->index + 1, buffer, 10);
|
_itow_s(child->index + 1, buffer, 10);
|
||||||
temp += buffer;
|
temp += buffer;
|
||||||
temp += L".ico";
|
temp += L".ico";
|
||||||
child->iconPath = RmReadPath(rm, L"IconPath", temp.c_str());
|
child->iconPath = RmReadPath(rm, L"IconPath", temp.c_str());
|
||||||
|
|
||||||
LPCWSTR size = RmReadString(rm, L"IconSize", L"MEDIUM");
|
LPCWSTR size = RmReadString(rm, L"IconSize", L"MEDIUM");
|
||||||
@ -258,6 +270,12 @@ PLUGIN_EXPORT void Reload(void* data, void* rm, double* maxValue)
|
|||||||
{
|
{
|
||||||
child->iconSize = IS_EXLARGE;
|
child->iconSize = IS_EXLARGE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
auto iter = std::find(child->parent->iconChildren.begin(), child->parent->iconChildren.end(), child);
|
||||||
|
if (iter == child->parent->iconChildren.end())
|
||||||
|
{
|
||||||
|
child->parent->iconChildren.push_back(child);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else if (_wcsicmp(type, L"FILEPATH") == 0)
|
else if (_wcsicmp(type, L"FILEPATH") == 0)
|
||||||
{
|
{
|
||||||
@ -279,22 +297,22 @@ PLUGIN_EXPORT double Update(void* data)
|
|||||||
if (parent->ownerChild == child && (parent->needsUpdating || parent->needsIcons))
|
if (parent->ownerChild == child && (parent->needsUpdating || parent->needsIcons))
|
||||||
{
|
{
|
||||||
unsigned int id;
|
unsigned int id;
|
||||||
HANDLE thread = (HANDLE)_beginthreadex(NULL, 0, SystemThreadProc, child, 0, &id);
|
HANDLE thread = (HANDLE)_beginthreadex(nullptr, 0, SystemThreadProc, parent, 0, &id);
|
||||||
if (thread)
|
if (thread)
|
||||||
{
|
{
|
||||||
parent->thread = thread;
|
parent->thread = thread;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int trueIndex = child->ignoreCount ? child->index : ((child->index % parent->count) + parent->indexOffset);
|
int trueIndex = child->ignoreCount ? child->index : ((child->index % parent->count) + parent->indexOffset);
|
||||||
child->value = 0;
|
double value = 0;
|
||||||
|
|
||||||
if (!parent->files.empty() && trueIndex >= 0 && trueIndex < parent->files.size())
|
if (!parent->files.empty() && trueIndex >= 0 && trueIndex < parent->files.size())
|
||||||
{
|
{
|
||||||
switch (child->type)
|
switch (child->type)
|
||||||
{
|
{
|
||||||
case TYPE_FILESIZE:
|
case TYPE_FILESIZE:
|
||||||
child->value = parent->files[trueIndex].size > 0 ? (double)parent->files[trueIndex].size : 0;
|
value = parent->files[trueIndex].size > 0 ? (double)parent->files[trueIndex].size : 0;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case TYPE_FILEDATE:
|
case TYPE_FILEDATE:
|
||||||
@ -319,42 +337,42 @@ PLUGIN_EXPORT double Update(void* data)
|
|||||||
}
|
}
|
||||||
|
|
||||||
FileTimeToSystemTime(&fTime, &stUTC);
|
FileTimeToSystemTime(&fTime, &stUTC);
|
||||||
SystemTimeToTzSpecificLocalTime(NULL, &stUTC, &stLOCAL);
|
SystemTimeToTzSpecificLocalTime(nullptr, &stUTC, &stLOCAL);
|
||||||
SystemTimeToFileTime(&stLOCAL, &fTime);
|
SystemTimeToFileTime(&stLOCAL, &fTime);
|
||||||
|
|
||||||
time.LowPart = fTime.dwLowDateTime;
|
time.LowPart = fTime.dwLowDateTime;
|
||||||
time.HighPart = fTime.dwHighDateTime;
|
time.HighPart = fTime.dwHighDateTime;
|
||||||
|
|
||||||
child->value = (double)(time.QuadPart / 10000000);
|
value = (double)(time.QuadPart / 10000000);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
switch (child->type)
|
switch (child->type)
|
||||||
{
|
{
|
||||||
case TYPE_FILECOUNT:
|
case TYPE_FILECOUNT:
|
||||||
child->value = (double)parent->fileCount;
|
value = (double)parent->fileCount;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case TYPE_FOLDERCOUNT:
|
case TYPE_FOLDERCOUNT:
|
||||||
child->value = (double)parent->folderCount;
|
value = (double)parent->folderCount;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case TYPE_FOLDERSIZE:
|
case TYPE_FOLDERSIZE:
|
||||||
child->value = (double)parent->folderSize;
|
value = (double)parent->folderSize;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
LeaveCriticalSection(&g_CriticalSection);
|
LeaveCriticalSection(&g_CriticalSection);
|
||||||
|
|
||||||
return child->value;
|
return value;
|
||||||
}
|
}
|
||||||
|
|
||||||
PLUGIN_EXPORT LPCWSTR GetString(void* data)
|
PLUGIN_EXPORT LPCWSTR GetString(void* data)
|
||||||
{
|
{
|
||||||
ChildMeasure* child = (ChildMeasure*)data;
|
ChildMeasure* child = (ChildMeasure*)data;
|
||||||
ParentMeasure* parent = child->parent;
|
ParentMeasure* parent = child->parent;
|
||||||
|
|
||||||
EnterCriticalSection(&g_CriticalSection);
|
EnterCriticalSection(&g_CriticalSection);
|
||||||
if (!parent)
|
if (!parent)
|
||||||
{
|
{
|
||||||
@ -373,7 +391,7 @@ PLUGIN_EXPORT LPCWSTR GetString(void* data)
|
|||||||
if (!parent->files[trueIndex].isFolder)
|
if (!parent->files[trueIndex].isFolder)
|
||||||
{
|
{
|
||||||
LeaveCriticalSection(&g_CriticalSection);
|
LeaveCriticalSection(&g_CriticalSection);
|
||||||
return NULL; // Force a numeric return (see the Update function)
|
return nullptr; // Force a numeric return (see the Update function)
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -423,11 +441,11 @@ PLUGIN_EXPORT LPCWSTR GetString(void* data)
|
|||||||
if (fTime.dwLowDateTime != 0 && fTime.dwHighDateTime != 0)
|
if (fTime.dwLowDateTime != 0 && fTime.dwHighDateTime != 0)
|
||||||
{
|
{
|
||||||
FileTimeToSystemTime(&fTime, &stUTC);
|
FileTimeToSystemTime(&fTime, &stUTC);
|
||||||
SystemTimeToTzSpecificLocalTime(NULL, &stUTC, &stLOCAL);
|
SystemTimeToTzSpecificLocalTime(nullptr, &stUTC, &stLOCAL);
|
||||||
GetDateFormat(LOCALE_USER_DEFAULT, 0, &stLOCAL, NULL, temp, MAX_LINE_LENGTH);
|
GetDateFormat(LOCALE_USER_DEFAULT, 0, &stLOCAL, nullptr, temp, MAX_LINE_LENGTH);
|
||||||
child->strValue = temp;
|
child->strValue = temp;
|
||||||
child->strValue += L" ";
|
child->strValue += L" ";
|
||||||
GetTimeFormat(LOCALE_USER_DEFAULT, 0, &stLOCAL, NULL, temp, MAX_LINE_LENGTH);
|
GetTimeFormat(LOCALE_USER_DEFAULT, 0, &stLOCAL, nullptr, temp, MAX_LINE_LENGTH);
|
||||||
child->strValue += temp;
|
child->strValue += temp;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -443,7 +461,7 @@ PLUGIN_EXPORT LPCWSTR GetString(void* data)
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case TYPE_FILEPATH:
|
case TYPE_FILEPATH:
|
||||||
child->strValue = (_wcsicmp(parent->files[trueIndex].fileName.c_str(), L"..") == 0) ? parent->path : parent->path + parent->files[trueIndex].fileName;
|
child->strValue = (_wcsicmp(parent->files[trueIndex].fileName.c_str(), L"..") == 0) ? parent->path : parent->files[trueIndex].path + parent->files[trueIndex].fileName;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -453,9 +471,8 @@ PLUGIN_EXPORT LPCWSTR GetString(void* data)
|
|||||||
case TYPE_FILECOUNT:
|
case TYPE_FILECOUNT:
|
||||||
case TYPE_FOLDERCOUNT:
|
case TYPE_FOLDERCOUNT:
|
||||||
case TYPE_FOLDERSIZE:
|
case TYPE_FOLDERSIZE:
|
||||||
child->strValue = L"";
|
|
||||||
LeaveCriticalSection(&g_CriticalSection);
|
LeaveCriticalSection(&g_CriticalSection);
|
||||||
return NULL; // Force numeric return (see the Update function)
|
return nullptr; // Force numeric return (see the Update function)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case TYPE_FOLDERPATH:
|
case TYPE_FOLDERPATH:
|
||||||
@ -478,7 +495,7 @@ PLUGIN_EXPORT void ExecuteBang(void* data, LPCWSTR args)
|
|||||||
LeaveCriticalSection(&g_CriticalSection);
|
LeaveCriticalSection(&g_CriticalSection);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (parent->ownerChild == child)
|
if (parent->ownerChild == child)
|
||||||
{
|
{
|
||||||
if (parent->files.size() > parent->count)
|
if (parent->files.size() > parent->count)
|
||||||
@ -536,14 +553,14 @@ PLUGIN_EXPORT void ExecuteBang(void* data, LPCWSTR args)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (_wcsicmp(args, L"UPDATE") == 0)
|
if (_wcsicmp(args, L"UPDATE") == 0)
|
||||||
{
|
{
|
||||||
parent->indexOffset = 0;
|
parent->indexOffset = 0;
|
||||||
parent->needsIcons = true;
|
parent->needsIcons = true;
|
||||||
parent->needsUpdating = true;
|
parent->needsUpdating = true;
|
||||||
}
|
}
|
||||||
else if (_wcsicmp(args, L"PREVIOUSFOLDER") == 0)
|
else if (parent->recursiveType != RECURSIVE_FULL && _wcsicmp(args, L"PREVIOUSFOLDER") == 0)
|
||||||
{
|
{
|
||||||
std::vector<std::wstring> path = Tokenize(parent->path, L"\\");
|
std::vector<std::wstring> path = Tokenize(parent->path, L"\\");
|
||||||
if (path.size() < 2)
|
if (path.size() < 2)
|
||||||
@ -571,13 +588,13 @@ PLUGIN_EXPORT void ExecuteBang(void* data, LPCWSTR args)
|
|||||||
|
|
||||||
int trueIndex = child->ignoreCount ? child->index : ((child->index % parent->count) + parent->indexOffset);
|
int trueIndex = child->ignoreCount ? child->index : ((child->index % parent->count) + parent->indexOffset);
|
||||||
if (!parent->files.empty() && trueIndex >= 0 && trueIndex < parent->files.size())
|
if (!parent->files.empty() && trueIndex >= 0 && trueIndex < parent->files.size())
|
||||||
{
|
{
|
||||||
if (_wcsicmp(args, L"OPEN") == 0)
|
if (_wcsicmp(args, L"OPEN") == 0)
|
||||||
{
|
{
|
||||||
std::wstring file = parent->path + parent->files[trueIndex].fileName;
|
std::wstring file = parent->files[trueIndex].path + parent->files[trueIndex].fileName;
|
||||||
ShellExecute(NULL, NULL, file.c_str(), NULL, parent->path.c_str(), SW_SHOW);
|
ShellExecute(nullptr, nullptr, file.c_str(), nullptr, parent->files[trueIndex].path.c_str(), SW_SHOW);
|
||||||
}
|
}
|
||||||
else if (_wcsicmp(args, L"FOLLOWPATH") == 0)
|
else if (parent->recursiveType != RECURSIVE_FULL && _wcsicmp(args, L"FOLLOWPATH") == 0)
|
||||||
{
|
{
|
||||||
if (_wcsicmp(parent->files[trueIndex].fileName.c_str(), L"..") == 0)
|
if (_wcsicmp(parent->files[trueIndex].fileName.c_str(), L"..") == 0)
|
||||||
{
|
{
|
||||||
@ -615,7 +632,7 @@ PLUGIN_EXPORT void ExecuteBang(void* data, LPCWSTR args)
|
|||||||
else
|
else
|
||||||
{
|
{
|
||||||
std::wstring file = parent->path + parent->files[trueIndex].fileName;
|
std::wstring file = parent->path + parent->files[trueIndex].fileName;
|
||||||
ShellExecute(NULL, NULL, file.c_str(), NULL, parent->path.c_str(), SW_SHOW);
|
ShellExecute(nullptr, nullptr, file.c_str(), nullptr, parent->path.c_str(), SW_SHOW);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -636,13 +653,13 @@ PLUGIN_EXPORT void Finalize(void* data)
|
|||||||
if (parent->thread)
|
if (parent->thread)
|
||||||
{
|
{
|
||||||
TerminateThread(parent->thread, 0);
|
TerminateThread(parent->thread, 0);
|
||||||
parent->thread = NULL;
|
parent->thread = nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (parent && parent->ownerChild == child)
|
if (parent && parent->ownerChild == child)
|
||||||
{
|
{
|
||||||
CloseHandle(parent->thread);
|
CloseHandle(parent->thread);
|
||||||
parent->thread = NULL;
|
parent->thread = nullptr;
|
||||||
|
|
||||||
delete parent;
|
delete parent;
|
||||||
|
|
||||||
@ -656,52 +673,29 @@ PLUGIN_EXPORT void Finalize(void* data)
|
|||||||
|
|
||||||
unsigned __stdcall SystemThreadProc(void* pParam)
|
unsigned __stdcall SystemThreadProc(void* pParam)
|
||||||
{
|
{
|
||||||
ChildMeasure* child = (ChildMeasure*)pParam;
|
ParentMeasure* parent = (ParentMeasure*)pParam;
|
||||||
ParentMeasure* parent = child->parent;
|
|
||||||
|
|
||||||
EnterCriticalSection(&g_CriticalSection);
|
EnterCriticalSection(&g_CriticalSection);
|
||||||
const bool needsUpdating = parent->needsUpdating;
|
ParentMeasure* tmp = new ParentMeasure (*parent);
|
||||||
parent->needsUpdating = false; // Set to false here in case skin is reloaded
|
parent->needsUpdating = false; // Set to false here in case skin is reloaded
|
||||||
const bool needsIcons = parent->needsIcons;
|
|
||||||
parent->needsIcons = false; // Set to false here in case skin is reloaded
|
parent->needsIcons = false; // Set to false here in case skin is reloaded
|
||||||
|
|
||||||
const std::wstring parentPath = parent->path;
|
|
||||||
const std::wstring wildcard = parent->wildcardSearch;
|
|
||||||
const std::vector<std::wstring> extensions = parent->extensions;
|
|
||||||
const bool isRecursive = parent->isRecursive;
|
|
||||||
const bool showDotDot = parent->showDotDot;
|
|
||||||
const bool showFile = parent->showFile;
|
|
||||||
const bool showFolder = parent->showFolder;
|
|
||||||
const bool showHidden = parent->showHidden;
|
|
||||||
const bool showSystem = parent->showSystem;
|
|
||||||
const SortType sortType = parent->sortType;
|
|
||||||
const DateType sortDateType = parent->sortDateType;
|
|
||||||
const bool sortAscending = parent->sortAscending;
|
|
||||||
const std::vector<ChildMeasure*> children = parent->children;
|
|
||||||
|
|
||||||
std::vector<FileInfo> files = parent->files;
|
|
||||||
int fileCount = parent->fileCount;
|
|
||||||
int folderCount = parent->folderCount;
|
|
||||||
UINT64 folderSize = parent->folderSize;
|
|
||||||
LeaveCriticalSection(&g_CriticalSection);
|
LeaveCriticalSection(&g_CriticalSection);
|
||||||
|
|
||||||
std::wstring path = parentPath + wildcard;
|
|
||||||
|
|
||||||
FileInfo file;
|
FileInfo file;
|
||||||
|
|
||||||
if (needsUpdating)
|
if (tmp->needsUpdating)
|
||||||
{
|
{
|
||||||
fileCount = 0;
|
|
||||||
folderCount = 0;
|
|
||||||
folderSize = 0;
|
|
||||||
|
|
||||||
EnterCriticalSection(&g_CriticalSection);
|
EnterCriticalSection(&g_CriticalSection);
|
||||||
parent->files.clear();
|
parent->files.clear();
|
||||||
files.clear();
|
|
||||||
LeaveCriticalSection(&g_CriticalSection);
|
LeaveCriticalSection(&g_CriticalSection);
|
||||||
|
|
||||||
|
tmp->files.clear();
|
||||||
|
tmp->fileCount = 0;
|
||||||
|
tmp->folderCount = 0;
|
||||||
|
tmp->folderSize = 0;
|
||||||
|
|
||||||
// If no path is specified, get all the drives instead
|
// If no path is specified, get all the drives instead
|
||||||
if (parentPath == L"" || parentPath.empty())
|
if (tmp->path == L"" || tmp->path.empty())
|
||||||
{
|
{
|
||||||
WCHAR drive[4] = L" :\\";
|
WCHAR drive[4] = L" :\\";
|
||||||
DWORD driveMask = GetLogicalDrives();
|
DWORD driveMask = GetLogicalDrives();
|
||||||
@ -713,241 +707,295 @@ unsigned __stdcall SystemThreadProc(void* pParam)
|
|||||||
file.fileName = drive;
|
file.fileName = drive;
|
||||||
file.isFolder = true;
|
file.isFolder = true;
|
||||||
file.size = 0;
|
file.size = 0;
|
||||||
++folderCount;
|
|
||||||
|
|
||||||
files.push_back(file);
|
++tmp->folderCount;
|
||||||
|
tmp->files.push_back(file);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if (showDotDot)
|
if (tmp->showDotDot && tmp->recursiveType != RECURSIVE_FULL)
|
||||||
{
|
{
|
||||||
file.fileName = L"..";
|
file.fileName = L"..";
|
||||||
file.isFolder = true;
|
file.isFolder = true;
|
||||||
|
|
||||||
files.push_back(file);
|
tmp->files.push_back(file);
|
||||||
}
|
}
|
||||||
|
|
||||||
WIN32_FIND_DATA fd;
|
std::queue<std::wstring> folderQueue;
|
||||||
HANDLE find = FindFirstFileEx(path.c_str(), FindExInfoStandard, &fd, FindExSearchNameMatch, NULL, 0);
|
std::wstring folder = tmp->path;
|
||||||
|
|
||||||
|
RecursiveType rType = tmp->recursiveType;
|
||||||
|
GetFolderInfo(folderQueue, folder, tmp, (rType == RECURSIVE_PARTIAL) ? RECURSIVE_NONE : rType);
|
||||||
|
|
||||||
if (find != INVALID_HANDLE_VALUE)
|
if (rType != RECURSIVE_NONE)
|
||||||
{
|
{
|
||||||
do
|
while (!folderQueue.empty())
|
||||||
{
|
|
||||||
file.Clear();
|
|
||||||
|
|
||||||
file.fileName = fd.cFileName;
|
|
||||||
if (_wcsicmp(file.fileName.c_str(), L".") == 0 || _wcsicmp(file.fileName.c_str(), L"..") == 0)
|
|
||||||
{
|
{
|
||||||
continue;
|
folder = folderQueue.front();
|
||||||
|
GetFolderInfo(folderQueue, folder, tmp, rType);
|
||||||
|
folderQueue.pop();
|
||||||
}
|
}
|
||||||
|
|
||||||
file.isFolder = (fd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) > 0;
|
|
||||||
bool isHidden = (fd.dwFileAttributes & FILE_ATTRIBUTE_HIDDEN) > 0;
|
|
||||||
bool isSystem = (fd.dwFileAttributes & FILE_ATTRIBUTE_SYSTEM) > 0;
|
|
||||||
|
|
||||||
if ((!showFile && !file.isFolder) || (!showFolder && file.isFolder) ||
|
|
||||||
(!showHidden && isHidden) || (!showSystem && isSystem))
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!file.isFolder)
|
|
||||||
{
|
|
||||||
size_t pos = file.fileName.find_last_of(L".");
|
|
||||||
if (pos != file.fileName.npos)
|
|
||||||
{
|
|
||||||
file.ext = file.fileName.substr(pos + 1);
|
|
||||||
|
|
||||||
if (extensions.size() > 0)
|
|
||||||
{
|
|
||||||
bool found = false;
|
|
||||||
for (auto iter = extensions.begin(); iter != extensions.end(); ++iter)
|
|
||||||
{
|
|
||||||
if (_wcsicmp((*iter).c_str(), file.ext.c_str()) == 0)
|
|
||||||
{
|
|
||||||
found = true;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!found)
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (extensions.size() > 0)
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (file.isFolder)
|
|
||||||
{
|
|
||||||
++folderCount;
|
|
||||||
file.size = 0;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
++fileCount;
|
|
||||||
file.size = ((UINT64)fd.nFileSizeHigh << 32) + fd.nFileSizeLow;
|
|
||||||
file.createdTime = fd.ftCreationTime;
|
|
||||||
file.modifiedTime = fd.ftLastWriteTime;
|
|
||||||
file.accessedTime = fd.ftLastAccessTime;
|
|
||||||
}
|
|
||||||
|
|
||||||
file.sortAscending = sortAscending; // Used in sort functions
|
|
||||||
folderSize += file.size;
|
|
||||||
|
|
||||||
files.push_back(file);
|
|
||||||
}
|
|
||||||
while (FindNextFile(find, &fd));
|
|
||||||
FindClose(find);
|
|
||||||
|
|
||||||
// Sort
|
|
||||||
auto begin = showDotDot ? files.begin() + 1: files.begin();
|
|
||||||
switch (sortType)
|
|
||||||
{
|
|
||||||
case STYPE_NAME:
|
|
||||||
std::sort(begin, files.end(), SortByName);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case STYPE_SIZE:
|
|
||||||
std::sort(begin, files.end(), SortBySize);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case STYPE_TYPE:
|
|
||||||
std::sort(begin, files.end(), SortByExtension);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case STYPE_DATE:
|
|
||||||
switch (sortDateType)
|
|
||||||
{
|
|
||||||
case DTYPE_MODIFIED:
|
|
||||||
std::sort(begin, files.end(), SortByModifiedTime);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case DTYPE_CREATED:
|
|
||||||
std::sort(begin, files.end(), SortByCreatedTime);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case DTYPE_ACCESSED:
|
|
||||||
std::sort(begin, files.end(), SortByAccessedTime);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isRecursive)
|
|
||||||
{
|
|
||||||
GetSubFolderSize(files, parentPath, folderCount, fileCount, folderSize);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Sort
|
||||||
|
const int sortAsc = tmp->sortAscending ? 1 : -1;
|
||||||
|
auto begin = ((tmp->path != L"" || !tmp->path.empty()) && (tmp->showDotDot && tmp->recursiveType != RECURSIVE_FULL)) ?
|
||||||
|
tmp->files.begin() + 1: tmp->files.begin();
|
||||||
|
|
||||||
|
switch (tmp->sortType)
|
||||||
|
{
|
||||||
|
case STYPE_NAME:
|
||||||
|
std::sort(begin, tmp->files.end(),
|
||||||
|
[&sortAsc](const FileInfo& file1, const FileInfo& file2) -> bool
|
||||||
|
{
|
||||||
|
if (file1.isFolder && file2.isFolder)
|
||||||
|
{
|
||||||
|
return (sortAsc * _wcsicmp(file1.fileName.c_str(), file2.fileName.c_str()) < 0);
|
||||||
|
}
|
||||||
|
else if (!file1.isFolder && !file2.isFolder)
|
||||||
|
{
|
||||||
|
return (sortAsc * _wcsicmp(file1.fileName.c_str(), file2.fileName.c_str()) < 0);
|
||||||
|
}
|
||||||
|
return file1.isFolder;
|
||||||
|
});
|
||||||
|
break;
|
||||||
|
|
||||||
|
case STYPE_SIZE:
|
||||||
|
std::sort(begin, tmp->files.end(),
|
||||||
|
[&sortAsc](const FileInfo& file1, const FileInfo& file2) -> bool
|
||||||
|
{
|
||||||
|
if (file1.isFolder && file2.isFolder)
|
||||||
|
{
|
||||||
|
return (sortAsc * _wcsicmp(file1.fileName.c_str(), file2.fileName.c_str()) < 0);
|
||||||
|
}
|
||||||
|
else if (!file1.isFolder && !file2.isFolder)
|
||||||
|
{
|
||||||
|
return (sortAsc > 0) ? (file1.size < file2.size) : (file1.size > file2.size);
|
||||||
|
}
|
||||||
|
return file1.isFolder;
|
||||||
|
});
|
||||||
|
break;
|
||||||
|
|
||||||
|
case STYPE_TYPE:
|
||||||
|
std::sort(begin, tmp->files.end(),
|
||||||
|
[&sortAsc](const FileInfo& file1, const FileInfo& file2) -> bool
|
||||||
|
{
|
||||||
|
if (file1.isFolder && file2.isFolder)
|
||||||
|
{
|
||||||
|
return (sortAsc * _wcsicmp(file1.fileName.c_str(), file2.fileName.c_str()) < 0);
|
||||||
|
}
|
||||||
|
else if (!file1.isFolder && !file2.isFolder)
|
||||||
|
{
|
||||||
|
int result = (file1.ext.empty() && file2.ext.empty()) ? 0 : sortAsc * _wcsicmp(file1.ext.c_str(), file2.ext.c_str());
|
||||||
|
return (0 != result) ? (result < 0) : (sortAsc * _wcsicmp(file1.fileName.c_str(), file2.fileName.c_str()) < 0);
|
||||||
|
}
|
||||||
|
return file1.isFolder;
|
||||||
|
});
|
||||||
|
break;
|
||||||
|
|
||||||
|
case STYPE_DATE:
|
||||||
|
switch (tmp->sortDateType)
|
||||||
|
{
|
||||||
|
case DTYPE_MODIFIED:
|
||||||
|
std::sort(begin, tmp->files.end(),
|
||||||
|
[&sortAsc](const FileInfo& file1, const FileInfo& file2) -> bool
|
||||||
|
{
|
||||||
|
if (file1.isFolder && file2.isFolder)
|
||||||
|
{
|
||||||
|
return (sortAsc * CompareFileTime(&file1.modifiedTime, &file2.modifiedTime) < 0);
|
||||||
|
}
|
||||||
|
else if (!file1.isFolder && !file2.isFolder)
|
||||||
|
{
|
||||||
|
return (sortAsc * CompareFileTime(&file1.modifiedTime, &file2.modifiedTime) < 0);
|
||||||
|
}
|
||||||
|
return file1.isFolder;
|
||||||
|
});
|
||||||
|
break;
|
||||||
|
|
||||||
|
case DTYPE_CREATED:
|
||||||
|
std::sort(begin, tmp->files.end(),
|
||||||
|
[&sortAsc](const FileInfo& file1, const FileInfo& file2) -> bool
|
||||||
|
{
|
||||||
|
if (file1.isFolder && file2.isFolder)
|
||||||
|
{
|
||||||
|
return (sortAsc * CompareFileTime(&file1.createdTime, &file2.createdTime) < 0);
|
||||||
|
}
|
||||||
|
else if (!file1.isFolder && !file2.isFolder)
|
||||||
|
{
|
||||||
|
return (sortAsc * CompareFileTime(&file1.createdTime, &file2.createdTime) < 0);
|
||||||
|
}
|
||||||
|
return file1.isFolder;
|
||||||
|
});
|
||||||
|
break;
|
||||||
|
|
||||||
|
case DTYPE_ACCESSED:
|
||||||
|
std::sort(begin, tmp->files.end(),
|
||||||
|
[&sortAsc](const FileInfo& file1, const FileInfo& file2) -> bool
|
||||||
|
{
|
||||||
|
if (file1.isFolder && file2.isFolder)
|
||||||
|
{
|
||||||
|
return (sortAsc * CompareFileTime(&file1.accessedTime, &file2.accessedTime) < 0);
|
||||||
|
}
|
||||||
|
else if (!file1.isFolder && !file2.isFolder)
|
||||||
|
{
|
||||||
|
return (sortAsc * CompareFileTime(&file1.accessedTime, &file2.accessedTime) < 0);
|
||||||
|
}
|
||||||
|
return file1.isFolder;
|
||||||
|
});
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
EnterCriticalSection(&g_CriticalSection);
|
||||||
|
parent->files.reserve(tmp->files.size());
|
||||||
|
parent->files = tmp->files;
|
||||||
|
parent->fileCount = tmp->fileCount;
|
||||||
|
parent->folderCount = tmp->folderCount;
|
||||||
|
parent->folderSize = tmp->folderSize;
|
||||||
|
LeaveCriticalSection(&g_CriticalSection);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (needsIcons)
|
if (tmp->needsIcons)
|
||||||
{
|
{
|
||||||
for (auto iter = children.begin(); iter != children.end(); ++iter)
|
for (auto iter : tmp->iconChildren)
|
||||||
{
|
{
|
||||||
EnterCriticalSection(&g_CriticalSection);
|
EnterCriticalSection(&g_CriticalSection);
|
||||||
int trueIndex = (*iter)->ignoreCount ? (*iter)->index : (((*iter)->index % (*iter)->parent->count) + (*iter)->parent->indexOffset);
|
int trueIndex = iter->ignoreCount ? iter->index : ((iter->index % iter->parent->count) + iter->parent->indexOffset);
|
||||||
|
|
||||||
if ((*iter)->type == TYPE_ICON && trueIndex >= 0 && trueIndex < files.size())
|
if (iter->type == TYPE_ICON && trueIndex >= 0 && trueIndex < tmp->files.size())
|
||||||
{
|
{
|
||||||
std::wstring filePath = parentPath;
|
std::wstring filePath = tmp->files[trueIndex].path;
|
||||||
filePath += (files[trueIndex].fileName == L"..") ? L"" : files[trueIndex].fileName;
|
filePath += (tmp->files[trueIndex].fileName == L"..") ? L"" :tmp->files[trueIndex].fileName;
|
||||||
GetIcon(filePath, (*iter)->iconPath, (*iter)->iconSize);
|
GetIcon(filePath, iter->iconPath, iter->iconSize);
|
||||||
}
|
}
|
||||||
else if ((*iter)->type == TYPE_ICON)
|
else if (iter->type == TYPE_ICON)
|
||||||
{
|
{
|
||||||
GetIcon(INVALID_FILE, (*iter)->iconPath, (*iter)->iconSize);
|
GetIcon(INVALID_FILE, iter->iconPath, iter->iconSize);
|
||||||
}
|
}
|
||||||
LeaveCriticalSection(&g_CriticalSection);
|
LeaveCriticalSection(&g_CriticalSection);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
EnterCriticalSection(&g_CriticalSection);
|
EnterCriticalSection(&g_CriticalSection);
|
||||||
parent->files = files;
|
|
||||||
parent->fileCount = fileCount;
|
|
||||||
parent->folderCount = folderCount;
|
|
||||||
parent->folderSize = folderSize;
|
|
||||||
|
|
||||||
CloseHandle(parent->thread);
|
CloseHandle(parent->thread);
|
||||||
parent->thread = NULL;
|
parent->thread = nullptr;
|
||||||
LeaveCriticalSection(&g_CriticalSection);
|
LeaveCriticalSection(&g_CriticalSection);
|
||||||
|
|
||||||
if (!parent->finishAction.empty())
|
if (!tmp->finishAction.empty())
|
||||||
{
|
{
|
||||||
RmExecute(parent->skin, parent->finishAction.c_str());
|
RmExecute(tmp->skin, tmp->finishAction.c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
_endthreadex(0);
|
delete tmp;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void GetSubFolderSize(const std::vector<FileInfo> files, const std::wstring path, int& folderCount, int& fileCount, UINT64& folderSize)
|
void GetFolderInfo(std::queue<std::wstring>& folderQueue, std::wstring& folder, ParentMeasure* parent, RecursiveType rType)
|
||||||
{
|
{
|
||||||
std::list<std::wstring> folderQueue;
|
std::wstring path = folder;
|
||||||
std::wstring folder;
|
folder += (rType == RECURSIVE_PARTIAL) ? L"*" : parent->wildcardSearch;
|
||||||
|
|
||||||
// Get current folders first (if any)
|
WIN32_FIND_DATA fd;
|
||||||
for (auto iter = files.begin(); iter != files.end(); ++iter)
|
HANDLE find = FindFirstFileEx(folder.c_str(), FindExInfoStandard, &fd, FindExSearchNameMatch, nullptr, 0);
|
||||||
|
|
||||||
|
if (find != INVALID_HANDLE_VALUE)
|
||||||
{
|
{
|
||||||
if ((*iter).isFolder && _wcsicmp((*iter).fileName.c_str(), L"..") != 0)
|
do
|
||||||
{
|
{
|
||||||
folderQueue.push_back(path + (*iter).fileName);
|
FileInfo file;
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
while (!folderQueue.empty())
|
file.fileName = fd.cFileName;
|
||||||
{
|
if (_wcsicmp(file.fileName.c_str(), L".") == 0 || _wcsicmp(file.fileName.c_str(), L"..") == 0)
|
||||||
std::list<std::wstring>::reference ref = folderQueue.front();
|
|
||||||
folder = ref + L"\\*";
|
|
||||||
|
|
||||||
WIN32_FIND_DATA fd;
|
|
||||||
HANDLE find = FindFirstFileEx(folder.c_str(), FindExInfoStandard, &fd, FindExSearchNameMatch, NULL, 0);
|
|
||||||
|
|
||||||
if (find != INVALID_HANDLE_VALUE)
|
|
||||||
{
|
|
||||||
do
|
|
||||||
{
|
{
|
||||||
folder = fd.cFileName;
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
if (_wcsicmp(folder.c_str(), L".") == 0 || _wcsicmp(folder.c_str(), L"..") == 0)
|
file.isFolder = (fd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) > 0;
|
||||||
|
bool isHidden = (fd.dwFileAttributes & FILE_ATTRIBUTE_HIDDEN) > 0;
|
||||||
|
bool isSystem = (fd.dwFileAttributes & FILE_ATTRIBUTE_SYSTEM) > 0;
|
||||||
|
|
||||||
|
if ((rType != RECURSIVE_PARTIAL) &&
|
||||||
|
((rType != RECURSIVE_FULL && !parent->showFile && !file.isFolder) ||
|
||||||
|
(rType != RECURSIVE_FULL && !parent->showFolder && file.isFolder) ||
|
||||||
|
(!parent->showHidden && isHidden) ||(!parent->showSystem && isSystem)))
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rType != RECURSIVE_PARTIAL && !file.isFolder)
|
||||||
|
{
|
||||||
|
size_t pos = file.fileName.find_last_of(L".");
|
||||||
|
if (pos != std::wstring::npos)
|
||||||
|
{
|
||||||
|
file.ext = file.fileName.substr(pos + 1);
|
||||||
|
|
||||||
|
if (parent->extensions.size() > 0)
|
||||||
|
{
|
||||||
|
bool found = false;
|
||||||
|
for (auto iter : parent->extensions)
|
||||||
|
{
|
||||||
|
if (_wcsicmp(iter.c_str(), file.ext.c_str()) == 0)
|
||||||
|
{
|
||||||
|
found = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!found)
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (parent->extensions.size() > 0)
|
||||||
{
|
{
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((fd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) > 0)
|
|
||||||
{
|
|
||||||
++folderCount;
|
|
||||||
folderQueue.push_back(ref + L"\\" + folder);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
++fileCount;
|
|
||||||
folderSize += ((UINT64)fd.nFileSizeHigh << 32) + fd.nFileSizeLow;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
while (FindNextFile(find, &fd));
|
|
||||||
|
if (file.isFolder)
|
||||||
FindClose(find);
|
{
|
||||||
|
if (rType != RECURSIVE_FULL)
|
||||||
|
{
|
||||||
|
++parent->folderCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
folderQueue.push(path + file.fileName + L"\\");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
++parent->fileCount;
|
||||||
|
file.size = ((UINT64)fd.nFileSizeHigh << 32) + fd.nFileSizeLow;
|
||||||
|
}
|
||||||
|
|
||||||
|
parent->folderSize += file.size;
|
||||||
|
|
||||||
|
file.createdTime = fd.ftCreationTime;
|
||||||
|
file.modifiedTime = fd.ftLastWriteTime;
|
||||||
|
file.accessedTime = fd.ftLastAccessTime;
|
||||||
|
|
||||||
|
file.path = path;
|
||||||
|
|
||||||
|
if (rType == RECURSIVE_NONE || (rType == RECURSIVE_FULL && !file.isFolder))
|
||||||
|
{
|
||||||
|
parent->files.push_back(file);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
while (FindNextFile(find, &fd));
|
||||||
folderQueue.pop_front();
|
FindClose(find);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void GetIcon(std::wstring filePath, std::wstring iconPath, IconSize iconSize)
|
void GetIcon(std::wstring filePath, std::wstring iconPath, IconSize iconSize)
|
||||||
{
|
{
|
||||||
SHFILEINFO shFileInfo;
|
SHFILEINFO shFileInfo;
|
||||||
HICON icon = NULL;
|
HICON icon = nullptr;
|
||||||
HIMAGELIST* hImageList = NULL;
|
HIMAGELIST* hImageList = nullptr;
|
||||||
FILE* fp = NULL;
|
FILE* fp = nullptr;
|
||||||
|
|
||||||
// Special case for .url files
|
// Special case for .url files
|
||||||
if (filePath.size() > 3 && _wcsicmp(filePath.substr(filePath.size() - 4).c_str(), L".URL") == 0)
|
if (filePath.size() > 3 && _wcsicmp(filePath.substr(filePath.size() - 4).c_str(), L".URL") == 0)
|
||||||
@ -973,7 +1021,7 @@ void GetIcon(std::wstring filePath, std::wstring iconPath, IconSize iconSize)
|
|||||||
case IS_MEDIUM: size = 32; break;
|
case IS_MEDIUM: size = 32; break;
|
||||||
}
|
}
|
||||||
|
|
||||||
PrivateExtractIcons(file.c_str(), iconIndex, size, size, &icon, NULL, 1, LR_LOADTRANSPARENT);
|
PrivateExtractIcons(file.c_str(), iconIndex, size, size, &icon, nullptr, 1, LR_LOADTRANSPARENT);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -983,7 +1031,7 @@ void GetIcon(std::wstring filePath, std::wstring iconPath, IconSize iconSize)
|
|||||||
HKEY hKey;
|
HKEY hKey;
|
||||||
|
|
||||||
RegOpenKeyEx(HKEY_CLASSES_ROOT, L"http\\shell\\open\\command", 0, KEY_QUERY_VALUE, &hKey);
|
RegOpenKeyEx(HKEY_CLASSES_ROOT, L"http\\shell\\open\\command", 0, KEY_QUERY_VALUE, &hKey);
|
||||||
RegQueryValueEx(hKey, NULL, NULL, NULL, (LPBYTE)buffer, &size);
|
RegQueryValueEx(hKey, nullptr, nullptr, nullptr, (LPBYTE)buffer, &size);
|
||||||
RegCloseKey(hKey);
|
RegCloseKey(hKey);
|
||||||
|
|
||||||
//Strip quotes
|
//Strip quotes
|
||||||
@ -997,8 +1045,8 @@ void GetIcon(std::wstring filePath, std::wstring iconPath, IconSize iconSize)
|
|||||||
filePath = browser;
|
filePath = browser;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (icon == NULL)
|
if (icon == nullptr)
|
||||||
{
|
{
|
||||||
SHGetFileInfo(filePath.c_str(), 0, &shFileInfo, sizeof(shFileInfo), SHGFI_SYSICONINDEX);
|
SHGetFileInfo(filePath.c_str(), 0, &shFileInfo, sizeof(shFileInfo), SHGFI_SYSICONINDEX);
|
||||||
SHGetImageList(iconSize, IID_IImageList, (void**) &hImageList);
|
SHGetImageList(iconSize, IID_IImageList, (void**) &hImageList);
|
||||||
@ -1006,7 +1054,7 @@ void GetIcon(std::wstring filePath, std::wstring iconPath, IconSize iconSize)
|
|||||||
}
|
}
|
||||||
|
|
||||||
errno_t error = _wfopen_s(&fp, iconPath.c_str(), L"wb");
|
errno_t error = _wfopen_s(&fp, iconPath.c_str(), L"wb");
|
||||||
if (filePath == INVALID_FILE || icon == NULL || (error == 0 && !SaveIcon(icon, fp)))
|
if (filePath == INVALID_FILE || icon == nullptr || (error == 0 && !SaveIcon(icon, fp)))
|
||||||
{
|
{
|
||||||
fwrite(iconPath.c_str(), 1, 1, fp); // Clears previous icon
|
fwrite(iconPath.c_str(), 1, 1, fp); // Clears previous icon
|
||||||
fclose(fp);
|
fclose(fp);
|
||||||
@ -1020,7 +1068,7 @@ HRESULT SaveIcon(HICON hIcon, FILE* fp)
|
|||||||
ICONINFO iconInfo;
|
ICONINFO iconInfo;
|
||||||
BITMAP bmColor;
|
BITMAP bmColor;
|
||||||
BITMAP bmMask;
|
BITMAP bmMask;
|
||||||
if (!fp || NULL == hIcon || !GetIconInfo(hIcon, &iconInfo) ||
|
if (!fp || nullptr == hIcon || !GetIconInfo(hIcon, &iconInfo) ||
|
||||||
!GetObject(iconInfo.hbmColor, sizeof(bmColor), &bmColor) ||
|
!GetObject(iconInfo.hbmColor, sizeof(bmColor), &bmColor) ||
|
||||||
!GetObject(iconInfo.hbmMask, sizeof(bmMask), &bmMask))
|
!GetObject(iconInfo.hbmMask, sizeof(bmMask), &bmMask))
|
||||||
return false;
|
return false;
|
||||||
@ -1029,14 +1077,14 @@ HRESULT SaveIcon(HICON hIcon, FILE* fp)
|
|||||||
if (bmColor.bmBitsPixel != 16 && bmColor.bmBitsPixel != 32)
|
if (bmColor.bmBitsPixel != 16 && bmColor.bmBitsPixel != 32)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
HDC dc = GetDC(NULL);
|
HDC dc = GetDC(nullptr);
|
||||||
BYTE bmiBytes[sizeof(BITMAPINFOHEADER) + 256 * sizeof(RGBQUAD)];
|
BYTE bmiBytes[sizeof(BITMAPINFOHEADER) + 256 * sizeof(RGBQUAD)];
|
||||||
BITMAPINFO* bmi = (BITMAPINFO*)bmiBytes;
|
BITMAPINFO* bmi = (BITMAPINFO*)bmiBytes;
|
||||||
|
|
||||||
// color bits
|
// color bits
|
||||||
memset(bmi, 0, sizeof(BITMAPINFO));
|
memset(bmi, 0, sizeof(BITMAPINFO));
|
||||||
bmi->bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
|
bmi->bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
|
||||||
GetDIBits(dc, iconInfo.hbmColor, 0, bmColor.bmHeight, NULL, bmi, DIB_RGB_COLORS);
|
GetDIBits(dc, iconInfo.hbmColor, 0, bmColor.bmHeight, nullptr, bmi, DIB_RGB_COLORS);
|
||||||
int colorBytesCount = bmi->bmiHeader.biSizeImage;
|
int colorBytesCount = bmi->bmiHeader.biSizeImage;
|
||||||
BYTE* colorBits = new BYTE[colorBytesCount];
|
BYTE* colorBits = new BYTE[colorBytesCount];
|
||||||
GetDIBits(dc, iconInfo.hbmColor, 0, bmColor.bmHeight, colorBits, bmi, DIB_RGB_COLORS);
|
GetDIBits(dc, iconInfo.hbmColor, 0, bmColor.bmHeight, colorBits, bmi, DIB_RGB_COLORS);
|
||||||
@ -1044,12 +1092,12 @@ HRESULT SaveIcon(HICON hIcon, FILE* fp)
|
|||||||
// mask bits
|
// mask bits
|
||||||
memset(bmi, 0, sizeof(BITMAPINFO));
|
memset(bmi, 0, sizeof(BITMAPINFO));
|
||||||
bmi->bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
|
bmi->bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
|
||||||
GetDIBits(dc, iconInfo.hbmMask, 0, bmMask.bmHeight, NULL, bmi, DIB_RGB_COLORS);
|
GetDIBits(dc, iconInfo.hbmMask, 0, bmMask.bmHeight, nullptr, bmi, DIB_RGB_COLORS);
|
||||||
int maskBytesCount = bmi->bmiHeader.biSizeImage;
|
int maskBytesCount = bmi->bmiHeader.biSizeImage;
|
||||||
BYTE* maskBits = new BYTE[maskBytesCount];
|
BYTE* maskBits = new BYTE[maskBytesCount];
|
||||||
GetDIBits(dc, iconInfo.hbmMask, 0, bmMask.bmHeight, maskBits, bmi, DIB_RGB_COLORS);
|
GetDIBits(dc, iconInfo.hbmMask, 0, bmMask.bmHeight, maskBits, bmi, DIB_RGB_COLORS);
|
||||||
|
|
||||||
ReleaseDC(NULL, dc);
|
ReleaseDC(nullptr, dc);
|
||||||
|
|
||||||
// icon data
|
// icon data
|
||||||
BITMAPINFOHEADER bmihIcon;
|
BITMAPINFOHEADER bmihIcon;
|
||||||
|
@ -20,19 +20,16 @@
|
|||||||
|
|
||||||
enum MeasureType
|
enum MeasureType
|
||||||
{
|
{
|
||||||
TYPE_FOLDERPATH, // Current folder complete path
|
TYPE_FOLDERPATH,
|
||||||
TYPE_FOLDERSIZE, // Current folder size
|
TYPE_FOLDERSIZE,
|
||||||
|
TYPE_FILECOUNT,
|
||||||
TYPE_FILECOUNT, // Number of files of current folder
|
TYPE_FOLDERCOUNT,
|
||||||
TYPE_FOLDERCOUNT, // Number of sub-folders under the current folder
|
TYPE_FILENAME,
|
||||||
|
TYPE_FILETYPE,
|
||||||
TYPE_FILENAME, // Name of file
|
TYPE_FILESIZE,
|
||||||
TYPE_FILETYPE, // Type of file (ie "Text Document", not .txt)
|
TYPE_FILEDATE,
|
||||||
TYPE_FILESIZE, // Size of file
|
TYPE_FILEPATH,
|
||||||
TYPE_FILEDATE, // Date of file - Can be "Created Date", "Modified Date" etc.
|
TYPE_ICON
|
||||||
TYPE_FILEPATH, // Full path of the file
|
|
||||||
|
|
||||||
TYPE_ICON // Icon of file
|
|
||||||
};
|
};
|
||||||
|
|
||||||
enum DateType
|
enum DateType
|
||||||
@ -58,51 +55,45 @@ enum IconSize
|
|||||||
IS_EXLARGE = 4 // 256x256
|
IS_EXLARGE = 4 // 256x256
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum RecursiveType
|
||||||
|
{
|
||||||
|
RECURSIVE_NONE,
|
||||||
|
RECURSIVE_PARTIAL,
|
||||||
|
RECURSIVE_FULL
|
||||||
|
};
|
||||||
|
|
||||||
struct FileInfo
|
struct FileInfo
|
||||||
{
|
{
|
||||||
std::wstring fileName;
|
std::wstring fileName;
|
||||||
std::wstring typeName; // File type description
|
std::wstring path;
|
||||||
std::wstring ext;
|
std::wstring ext;
|
||||||
bool isFolder;
|
bool isFolder;
|
||||||
bool sortAscending; // Used for sorting function (since we cannot pass other values to a sort function)
|
|
||||||
UINT64 size;
|
UINT64 size;
|
||||||
FILETIME createdTime;
|
FILETIME createdTime;
|
||||||
FILETIME modifiedTime;
|
FILETIME modifiedTime;
|
||||||
FILETIME accessedTime;
|
FILETIME accessedTime;
|
||||||
|
|
||||||
FileInfo(): sortAscending(false)
|
FileInfo() :
|
||||||
{
|
fileName(L""),
|
||||||
Clear();
|
path(L""),
|
||||||
}
|
ext(L""),
|
||||||
|
isFolder(false),
|
||||||
void Clear()
|
size(0),
|
||||||
{
|
createdTime(),
|
||||||
fileName = L"";
|
modifiedTime(),
|
||||||
typeName = L"";
|
accessedTime() { }
|
||||||
ext = L"";
|
|
||||||
isFolder = false;
|
|
||||||
size = 0;
|
|
||||||
|
|
||||||
createdTime.dwLowDateTime = 0;
|
|
||||||
createdTime.dwHighDateTime = 0;
|
|
||||||
modifiedTime.dwLowDateTime = 0;
|
|
||||||
modifiedTime.dwHighDateTime = 0;
|
|
||||||
accessedTime.dwLowDateTime = 0;
|
|
||||||
accessedTime.dwHighDateTime = 0;
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
struct ChildMeasure;
|
struct ChildMeasure;
|
||||||
|
|
||||||
struct ParentMeasure
|
struct ParentMeasure
|
||||||
{
|
{
|
||||||
// Options from the .ini
|
|
||||||
std::wstring path;
|
std::wstring path;
|
||||||
std::wstring wildcardSearch;
|
std::wstring wildcardSearch;
|
||||||
SortType sortType;
|
SortType sortType;
|
||||||
DateType sortDateType;
|
DateType sortDateType;
|
||||||
int count;
|
int count;
|
||||||
bool isRecursive;
|
RecursiveType recursiveType;
|
||||||
bool sortAscending;
|
bool sortAscending;
|
||||||
bool showDotDot;
|
bool showDotDot;
|
||||||
bool showFile;
|
bool showFile;
|
||||||
@ -110,11 +101,10 @@ struct ParentMeasure
|
|||||||
bool showHidden;
|
bool showHidden;
|
||||||
bool showSystem;
|
bool showSystem;
|
||||||
bool hideExtension;
|
bool hideExtension;
|
||||||
std::vector<std::wstring> extensions; // only show these extensions
|
std::vector<std::wstring> extensions;
|
||||||
std::wstring finishAction;
|
std::wstring finishAction;
|
||||||
|
|
||||||
// Internal values
|
std::vector<ChildMeasure*> iconChildren;
|
||||||
std::vector<ChildMeasure*> children;
|
|
||||||
std::vector<FileInfo> files;
|
std::vector<FileInfo> files;
|
||||||
int fileCount;
|
int fileCount;
|
||||||
int folderCount;
|
int folderCount;
|
||||||
@ -124,7 +114,6 @@ struct ParentMeasure
|
|||||||
int indexOffset;
|
int indexOffset;
|
||||||
HANDLE thread;
|
HANDLE thread;
|
||||||
|
|
||||||
// References and identifying values
|
|
||||||
void* skin;
|
void* skin;
|
||||||
LPCWSTR name;
|
LPCWSTR name;
|
||||||
ChildMeasure* ownerChild;
|
ChildMeasure* ownerChild;
|
||||||
@ -145,23 +134,22 @@ struct ParentMeasure
|
|||||||
hideExtension(false),
|
hideExtension(false),
|
||||||
extensions(),
|
extensions(),
|
||||||
finishAction(),
|
finishAction(),
|
||||||
children(NULL),
|
iconChildren(),
|
||||||
files(NULL),
|
files(),
|
||||||
skin(NULL),
|
skin(nullptr),
|
||||||
name(L""),
|
name(L""),
|
||||||
ownerChild(NULL),
|
ownerChild(nullptr),
|
||||||
thread(NULL),
|
thread(nullptr),
|
||||||
fileCount(0),
|
fileCount(0),
|
||||||
folderCount(0),
|
folderCount(0),
|
||||||
needsUpdating(true),
|
needsUpdating(true),
|
||||||
needsIcons(true),
|
needsIcons(true),
|
||||||
indexOffset(0),
|
indexOffset(0),
|
||||||
isRecursive(false) { }
|
recursiveType(RECURSIVE_NONE) { }
|
||||||
};
|
};
|
||||||
|
|
||||||
struct ChildMeasure
|
struct ChildMeasure
|
||||||
{
|
{
|
||||||
// Options from the .ini
|
|
||||||
MeasureType type;
|
MeasureType type;
|
||||||
DateType date;
|
DateType date;
|
||||||
IconSize iconSize;
|
IconSize iconSize;
|
||||||
@ -170,11 +158,7 @@ struct ChildMeasure
|
|||||||
bool ignoreCount;
|
bool ignoreCount;
|
||||||
bool needsIcon;
|
bool needsIcon;
|
||||||
|
|
||||||
// Internal values
|
std::wstring strValue;
|
||||||
double value; // numerical value of the value (if available)
|
|
||||||
std::wstring strValue; // string value of the value
|
|
||||||
|
|
||||||
// References
|
|
||||||
ParentMeasure* parent;
|
ParentMeasure* parent;
|
||||||
|
|
||||||
ChildMeasure() :
|
ChildMeasure() :
|
||||||
@ -185,125 +169,27 @@ struct ChildMeasure
|
|||||||
index(1),
|
index(1),
|
||||||
ignoreCount(false),
|
ignoreCount(false),
|
||||||
needsIcon(true),
|
needsIcon(true),
|
||||||
value(0.0),
|
strValue(L""),
|
||||||
strValue(),
|
parent(nullptr) { }
|
||||||
parent(NULL) { }
|
|
||||||
};
|
};
|
||||||
|
|
||||||
std::vector<std::wstring> Tokenize(const std::wstring& str, const std::wstring& delimiters)
|
std::vector<std::wstring> Tokenize(const std::wstring& str, const std::wstring& delimiters)
|
||||||
{
|
{
|
||||||
std::vector<std::wstring> tokens;
|
std::vector<std::wstring> tokens;
|
||||||
|
|
||||||
std::wstring::size_type lastPos = str.find_first_not_of(delimiters, 0); // skip delimiters at beginning.
|
std::wstring::size_type lastPos = str.find_first_not_of(delimiters, 0);
|
||||||
std::wstring::size_type pos = str.find_first_of(delimiters, lastPos); // find first "non-delimiter".
|
std::wstring::size_type pos = str.find_first_of(delimiters, lastPos);
|
||||||
|
|
||||||
while (std::wstring::npos != pos || std::wstring::npos != lastPos)
|
while (std::wstring::npos != pos || std::wstring::npos != lastPos)
|
||||||
{
|
{
|
||||||
tokens.push_back(str.substr(lastPos, pos - lastPos)); // found a token, add it to the vector.
|
tokens.emplace_back(str.substr(lastPos, pos - lastPos));
|
||||||
lastPos = str.find_first_not_of(delimiters, pos); // skip delimiters. Note the "not_of"
|
lastPos = str.find_first_not_of(delimiters, pos);
|
||||||
pos = str.find_first_of(delimiters, lastPos); // find next "non-delimiter"
|
pos = str.find_first_of(delimiters, lastPos);
|
||||||
}
|
}
|
||||||
|
|
||||||
return tokens;
|
return tokens;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool SortByName(const FileInfo& file1, const FileInfo& file2)
|
|
||||||
{
|
|
||||||
int sort = file1.sortAscending ? 1 : -1;
|
|
||||||
|
|
||||||
if (file1.isFolder && file2.isFolder)
|
|
||||||
{
|
|
||||||
return (sort * _wcsicmp(file1.fileName.c_str(), file2.fileName.c_str()) < 0);
|
|
||||||
}
|
|
||||||
else if (!file1.isFolder && !file2.isFolder)
|
|
||||||
{
|
|
||||||
return (sort * _wcsicmp(file1.fileName.c_str(), file2.fileName.c_str()) < 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
return file1.isFolder;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool SortByExtension(const FileInfo& file1, const FileInfo& file2)
|
|
||||||
{
|
|
||||||
int sort = file1.sortAscending ? 1 : -1;
|
|
||||||
|
|
||||||
if (file1.isFolder && file2.isFolder)
|
|
||||||
{
|
|
||||||
return (sort * _wcsicmp(file1.fileName.c_str(), file2.fileName.c_str()) < 0);
|
|
||||||
}
|
|
||||||
else if (!file1.isFolder && !file2.isFolder)
|
|
||||||
{
|
|
||||||
int result = (file1.ext.empty() && file2.ext.empty()) ? 0 : sort * _wcsicmp(file1.ext.c_str(), file2.ext.c_str());
|
|
||||||
return (0 != result) ? (result < 0) : (sort * _wcsicmp(file1.fileName.c_str(), file2.fileName.c_str()) < 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
return file1.isFolder;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool SortBySize(const FileInfo& file1, const FileInfo& file2)
|
|
||||||
{
|
|
||||||
int sort = file1.sortAscending ? 1 : -1;
|
|
||||||
|
|
||||||
if (file1.isFolder && file2.isFolder)
|
|
||||||
{
|
|
||||||
return (sort * _wcsicmp(file1.fileName.c_str(), file2.fileName.c_str()) < 0);
|
|
||||||
}
|
|
||||||
else if (!file1.isFolder && !file2.isFolder)
|
|
||||||
{
|
|
||||||
return (sort > 0) ? (file1.size < file2.size) : (file1.size > file2.size);
|
|
||||||
}
|
|
||||||
|
|
||||||
return file1.isFolder;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool SortByAccessedTime(const FileInfo& file1, const FileInfo& file2)
|
|
||||||
{
|
|
||||||
int sort = file1.sortAscending ? 1 : -1;
|
|
||||||
|
|
||||||
if (file1.isFolder && file2.isFolder)
|
|
||||||
{
|
|
||||||
return (sort * CompareFileTime(&file1.accessedTime, &file2.accessedTime) < 0);
|
|
||||||
}
|
|
||||||
else if (!file1.isFolder && !file2.isFolder)
|
|
||||||
{
|
|
||||||
return (sort * CompareFileTime(&file1.accessedTime, &file2.accessedTime) < 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
return file1.isFolder;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool SortByCreatedTime(const FileInfo& file1, const FileInfo& file2)
|
|
||||||
{
|
|
||||||
int sort = file1.sortAscending ? 1 : -1;
|
|
||||||
|
|
||||||
if (file1.isFolder && file2.isFolder)
|
|
||||||
{
|
|
||||||
return (sort * CompareFileTime(&file1.createdTime, &file2.createdTime) < 0);
|
|
||||||
}
|
|
||||||
else if (!file1.isFolder && !file2.isFolder)
|
|
||||||
{
|
|
||||||
return (sort * CompareFileTime(&file1.createdTime, &file2.createdTime) < 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
return file1.isFolder;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool SortByModifiedTime(const FileInfo& file1, const FileInfo& file2)
|
|
||||||
{
|
|
||||||
int sort = file1.sortAscending ? 1 : -1;
|
|
||||||
|
|
||||||
if (file1.isFolder && file2.isFolder)
|
|
||||||
{
|
|
||||||
return (sort * CompareFileTime(&file1.modifiedTime, &file2.modifiedTime) < 0);
|
|
||||||
}
|
|
||||||
else if (!file1.isFolder && !file2.isFolder)
|
|
||||||
{
|
|
||||||
return (sort * CompareFileTime(&file1.modifiedTime, &file2.modifiedTime) < 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
return file1.isFolder;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*std::wstring UINT64_To_String(UINT64 value)
|
/*std::wstring UINT64_To_String(UINT64 value)
|
||||||
{
|
{
|
||||||
std::wstring result;
|
std::wstring result;
|
||||||
|
@ -7,7 +7,7 @@
|
|||||||
//
|
//
|
||||||
|
|
||||||
VS_VERSION_INFO VERSIONINFO
|
VS_VERSION_INFO VERSIONINFO
|
||||||
FILEVERSION 1,2,0,0
|
FILEVERSION 2,0,0,0
|
||||||
PRODUCTVERSION PRODUCTVER
|
PRODUCTVERSION PRODUCTVER
|
||||||
FILEFLAGSMASK 0x17L
|
FILEFLAGSMASK 0x17L
|
||||||
#ifdef _DEBUG
|
#ifdef _DEBUG
|
||||||
@ -23,7 +23,7 @@ VS_VERSION_INFO VERSIONINFO
|
|||||||
{
|
{
|
||||||
BLOCK "040904E4"
|
BLOCK "040904E4"
|
||||||
{
|
{
|
||||||
VALUE "FileVersion", "1.2.0.0"
|
VALUE "FileVersion", "2.0.0.0"
|
||||||
VALUE "LegalCopyright", "© 2012 - Brian Ferguson"
|
VALUE "LegalCopyright", "© 2012 - Brian Ferguson"
|
||||||
VALUE "ProductName", "Rainmeter"
|
VALUE "ProductName", "Rainmeter"
|
||||||
#ifdef _WIN64
|
#ifdef _WIN64
|
||||||
|
@ -28,7 +28,7 @@
|
|||||||
// STL
|
// STL
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include <list>
|
#include <queue>
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
|
|
||||||
// Rainmeter API
|
// Rainmeter API
|
||||||
|
Loading…
x
Reference in New Issue
Block a user