mirror of
https://github.com/WinampDesktop/winamp.git
synced 2025-01-28 15:08:23 +00:00
940 lines
24 KiB
C++
940 lines
24 KiB
C++
|
#include "main.h"
|
||
|
#include "resource.h"
|
||
|
#include "api_mldb.h"
|
||
|
#include "../Winamp/strutil.h"
|
||
|
|
||
|
enum {
|
||
|
STATUS_SEARCHING,
|
||
|
STATUS_GETINFO,
|
||
|
STATUS_DONE,
|
||
|
};
|
||
|
|
||
|
extern HWND g_bgrescan_status_hwnd;
|
||
|
extern nde_scanner_t m_media_scanner;
|
||
|
|
||
|
#define MAX_RECURSE_DEPTH 32
|
||
|
/* Event handles */
|
||
|
static HANDLE scan_killswitch=0;
|
||
|
static HANDLE scan_cancel=0;
|
||
|
static HANDLE scan_cancel_complete=0;
|
||
|
/* Thread handle */
|
||
|
static HANDLE scan_thread=0;
|
||
|
/* extension list */
|
||
|
static wchar_t *scan_extlist=0;
|
||
|
|
||
|
static void SyncTable()
|
||
|
{
|
||
|
EnterCriticalSection(&g_db_cs);
|
||
|
NDE_Table_Sync(g_table);
|
||
|
g_table_dirty=0;
|
||
|
LeaveCriticalSection(&g_db_cs);
|
||
|
}
|
||
|
|
||
|
static bool ScanCancelled(HANDLE *events, int count)
|
||
|
{
|
||
|
// make sure no one cancelled us
|
||
|
DWORD eventFired=WaitForMultipleObjectsEx(count, events, FALSE, 0, TRUE);
|
||
|
if (eventFired >= WAIT_OBJECT_0 && eventFired < (WAIT_OBJECT_0+count))
|
||
|
return true;
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
static bool SupportedType(const wchar_t *ext)
|
||
|
{
|
||
|
if (!scan_extlist)
|
||
|
scan_extlist=(wchar_t*)SendMessage(plugin.hwndWinampParent,WM_WA_IPC,0,IPC_GET_EXTLISTW);
|
||
|
|
||
|
// dunno how this would happen but should verify
|
||
|
if (!scan_extlist || scan_extlist == (wchar_t *)1)
|
||
|
return false;
|
||
|
|
||
|
const wchar_t *a = scan_extlist;
|
||
|
while (a && *a)
|
||
|
{
|
||
|
if (CompareStringW(MAKELCID(MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US), SORT_DEFAULT), NORM_IGNORECASE, a, -1, ext, -1) == CSTR_EQUAL)
|
||
|
{
|
||
|
return true;
|
||
|
}
|
||
|
a+=wcslen(a)+1;
|
||
|
}
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
static void CountFolder(const wchar_t *folder, int recurse, HANDLE cancelswitch, volatile int *found)
|
||
|
{
|
||
|
wchar_t filespec[MAX_PATH] = {0};
|
||
|
PathCombineW(filespec, folder, L"*.*");
|
||
|
|
||
|
WIN32_FIND_DATAW findData = {0};
|
||
|
|
||
|
HANDLE h = FindFirstFileW(filespec, &findData);
|
||
|
if (h != INVALID_HANDLE_VALUE)
|
||
|
{
|
||
|
if (IsWindow(g_bgrescan_status_hwnd))
|
||
|
{
|
||
|
wchar_t status[150+MAX_PATH] = {0};
|
||
|
WASABI_API_LNGSTRINGW_BUF(IDS_SCANNING_DIR, status, 150);
|
||
|
|
||
|
const wchar_t *p=folder+wcslen(folder);
|
||
|
while (p > folder && *p != '\\') p--;
|
||
|
p--;
|
||
|
while (p >= folder && *p != '\\') p--;
|
||
|
|
||
|
StringCbCatW(status, sizeof(status), ++p);
|
||
|
|
||
|
SetWindowTextW(g_bgrescan_status_hwnd,status);
|
||
|
}
|
||
|
|
||
|
HANDLE events[2] = { scan_killswitch, cancelswitch};
|
||
|
do
|
||
|
{
|
||
|
// make sure no one cancelled us
|
||
|
if (ScanCancelled(events, 2))
|
||
|
break;
|
||
|
|
||
|
/* if it's a directory (And not either of the two special dirs */
|
||
|
if ((findData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
|
||
|
&& lstrcmpiW(findData.cFileName, L".")
|
||
|
&& lstrcmpiW(findData.cFileName, L".."))
|
||
|
{
|
||
|
if (recurse && recurse < MAX_RECURSE_DEPTH)
|
||
|
{
|
||
|
PathCombineW(filespec, folder, findData.cFileName);
|
||
|
CountFolder(filespec, recurse+1, cancelswitch, found); // add 1 so we can verify recurse depth
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (!(findData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY))
|
||
|
{
|
||
|
wchar_t *ext=extensionW(findData.cFileName);
|
||
|
if (ext && ext[0] && SupportedType(ext))
|
||
|
{
|
||
|
PathCombineW(filespec, folder, findData.cFileName);
|
||
|
if (IsWindow(g_bgrescan_status_hwnd))
|
||
|
{
|
||
|
wchar_t b[150+MAX_PATH] = {0};
|
||
|
WASABI_API_LNGSTRINGW_BUF(IDS_SCANNING_FILE, b, 150);
|
||
|
StringCbCatW(b, sizeof(b), filespec);
|
||
|
SetWindowTextW(g_bgrescan_status_hwnd,b);
|
||
|
}
|
||
|
if (found)
|
||
|
(*found)++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
} while (FindNextFileW(h, &findData));
|
||
|
FindClose(h);
|
||
|
}
|
||
|
else if (!(GetFileAttributesW(folder) & FILE_ATTRIBUTE_DIRECTORY))
|
||
|
{
|
||
|
if (IsWindow(g_bgrescan_status_hwnd))
|
||
|
{
|
||
|
wchar_t b[150+MAX_PATH] = {0};
|
||
|
WASABI_API_LNGSTRINGW_BUF(IDS_SCANNING_FILE, b, 150);
|
||
|
StringCbCatW(b, sizeof(b), folder);
|
||
|
SetWindowTextW(g_bgrescan_status_hwnd,b);
|
||
|
}
|
||
|
if (found)
|
||
|
(*found)++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void ScanFolder(const wchar_t *folder, int recurse, int metadata, int guessmode, HANDLE cancelswitch, volatile int *scanned)
|
||
|
{
|
||
|
if ((unsigned long)folder < 65536) return;
|
||
|
|
||
|
wchar_t filespec[MAX_PATH] = {0};
|
||
|
wchar_t status[150+MAX_PATH] = {0};
|
||
|
PathCombineW(filespec, folder, L"*.*");
|
||
|
|
||
|
WIN32_FIND_DATAW findData = {0};
|
||
|
|
||
|
HANDLE h = FindFirstFileW(filespec, &findData);
|
||
|
if (h != INVALID_HANDLE_VALUE)
|
||
|
{
|
||
|
if (IsWindow(g_bgrescan_status_hwnd))
|
||
|
{
|
||
|
WASABI_API_LNGSTRINGW_BUF(IDS_SCANNING_DIR, status, 150);
|
||
|
|
||
|
const wchar_t *p=folder+wcslen(folder);
|
||
|
while (p > folder && *p != '\\') p--;
|
||
|
p--;
|
||
|
while (p >= folder && *p != '\\') p--;
|
||
|
|
||
|
StringCbCatW(status, sizeof(status), ++p);
|
||
|
|
||
|
SetWindowTextW(g_bgrescan_status_hwnd,status);
|
||
|
}
|
||
|
|
||
|
HANDLE events[2] = { scan_killswitch, cancelswitch};
|
||
|
do
|
||
|
{
|
||
|
// make sure no one cancelled us
|
||
|
if (ScanCancelled(events, 2))
|
||
|
break;
|
||
|
|
||
|
/* if it's a directory (And not either of the two special dirs */
|
||
|
if ((findData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
|
||
|
&& lstrcmpiW(findData.cFileName, L".")
|
||
|
&& lstrcmpiW(findData.cFileName, L".."))
|
||
|
{
|
||
|
if (recurse && recurse < MAX_RECURSE_DEPTH)
|
||
|
{
|
||
|
PathCombineW(filespec, folder, findData.cFileName);
|
||
|
ScanFolder(filespec, recurse+1, metadata, guessmode, cancelswitch, scanned); // add 1 so we can verify recurse depth
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (!(findData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY))
|
||
|
{
|
||
|
wchar_t *ext=extensionW(findData.cFileName);
|
||
|
if (ext && ext[0] && SupportedType(ext))
|
||
|
{
|
||
|
PathCombineW(filespec, folder, findData.cFileName);
|
||
|
if (IsWindow(g_bgrescan_status_hwnd))
|
||
|
{
|
||
|
WASABI_API_LNGSTRINGW_BUF(IDS_SCANNING_FILE, status, 150);
|
||
|
StringCbCatW(status, sizeof(status), filespec);
|
||
|
SetWindowTextW(g_bgrescan_status_hwnd,status);
|
||
|
}
|
||
|
addFileToDb(filespec, 0, metadata, guessmode);
|
||
|
if (scanned)
|
||
|
(*scanned)++;
|
||
|
}
|
||
|
}
|
||
|
} while (FindNextFileW(h, &findData));
|
||
|
FindClose(h);
|
||
|
}
|
||
|
else if (!(GetFileAttributesW(folder) & FILE_ATTRIBUTE_DIRECTORY))
|
||
|
{
|
||
|
if (IsWindow(g_bgrescan_status_hwnd))
|
||
|
{
|
||
|
WASABI_API_LNGSTRINGW_BUF(IDS_SCANNING_FILE, status, 150);
|
||
|
StringCbCatW(status, sizeof(status), folder);
|
||
|
SetWindowTextW(g_bgrescan_status_hwnd,status);
|
||
|
}
|
||
|
addFileToDb(folder, 0, metadata, guessmode);
|
||
|
if (scanned)
|
||
|
(*scanned)++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static DWORD CALLBACK ScanThreadProc(LPVOID param)
|
||
|
{
|
||
|
/* sit and run APCs until we get signalled to die */
|
||
|
HANDLE events[2] = { scan_killswitch, scan_cancel};
|
||
|
int eventFired;
|
||
|
do
|
||
|
{
|
||
|
eventFired=WaitForMultipleObjectsEx(2, events, FALSE, INFINITE, TRUE);
|
||
|
switch(eventFired)
|
||
|
{
|
||
|
case WAIT_OBJECT_0+1: // cancel event
|
||
|
ResetEvent(scan_cancel);
|
||
|
SetEvent(scan_cancel_complete);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
while (eventFired != WAIT_OBJECT_0);
|
||
|
|
||
|
if (scan_extlist && scan_extlist != (wchar_t *)1)
|
||
|
GlobalFree((HGLOBAL)scan_extlist);
|
||
|
scan_extlist=0;
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static bool ScanCreateThread()
|
||
|
{
|
||
|
if (!scan_thread)
|
||
|
{
|
||
|
/* create events */
|
||
|
scan_killswitch = CreateEvent(NULL, TRUE, FALSE, NULL);
|
||
|
scan_cancel = CreateEvent(NULL, TRUE, FALSE, NULL);
|
||
|
scan_cancel_complete = CreateEvent(NULL, FALSE, FALSE, NULL); // auto-reset event
|
||
|
|
||
|
/* start thread */
|
||
|
scan_thread = CreateThread(NULL, 0, ScanThreadProc, 0, 0, 0);
|
||
|
}
|
||
|
|
||
|
return !!scan_thread;
|
||
|
}
|
||
|
|
||
|
void Scan_Cancel()
|
||
|
{
|
||
|
HWND old = g_bgrescan_status_hwnd; // clear g_bgrescan_status_hwnd so that we don't deadlock when the BG thread calls SetWindowText
|
||
|
g_bgrescan_status_hwnd = 0;
|
||
|
if (scan_cancel)
|
||
|
SignalObjectAndWait(scan_cancel, scan_cancel_complete, INFINITE, FALSE);
|
||
|
g_bgrescan_status_hwnd = old;
|
||
|
}
|
||
|
|
||
|
void Scan_Kill()
|
||
|
{
|
||
|
HWND old = g_bgrescan_status_hwnd; // clear g_bgrescan_status_hwnd so that we don't deadlock when the BG thread calls SetWindowText
|
||
|
g_bgrescan_status_hwnd = 0;
|
||
|
if (scan_thread)
|
||
|
SignalObjectAndWait(scan_killswitch, scan_thread, INFINITE, FALSE);
|
||
|
g_bgrescan_status_hwnd = old;
|
||
|
}
|
||
|
|
||
|
/* ---------------
|
||
|
* Scan_ScanFolder
|
||
|
* ---------------
|
||
|
*/
|
||
|
|
||
|
struct ScanFolderParams
|
||
|
{
|
||
|
ScanFolderParams(const wchar_t *_path, int _guess, int _meta, int _recurse)
|
||
|
{
|
||
|
path = _wcsdup(_path);
|
||
|
guess = _guess >= 0 ? _guess : g_config->ReadInt(L"guessmode",0);;
|
||
|
meta = _meta >= 0 ? _meta : g_config->ReadInt(L"usemetadata",1);
|
||
|
recurse = _recurse;
|
||
|
}
|
||
|
~ScanFolderParams()
|
||
|
{
|
||
|
free(path);
|
||
|
}
|
||
|
wchar_t *path;
|
||
|
int guess;
|
||
|
int meta;
|
||
|
int recurse;
|
||
|
};
|
||
|
|
||
|
static VOID CALLBACK ScanFolderAPC(ULONG_PTR param)
|
||
|
{
|
||
|
// clear extension list to get latest config
|
||
|
if (scan_extlist && scan_extlist != (wchar_t *)1)
|
||
|
GlobalFree((HGLOBAL)scan_extlist);
|
||
|
scan_extlist = 0;
|
||
|
|
||
|
ScanFolderParams *params = (ScanFolderParams *)param;
|
||
|
ScanFolder(params->path, params->recurse, params->meta, params->guess, scan_cancel, 0);
|
||
|
SyncTable();
|
||
|
delete params;
|
||
|
}
|
||
|
|
||
|
void Scan_ScanFolderBackground(const wchar_t *path, int guess, int meta, int recurse)
|
||
|
{
|
||
|
if (ScanCreateThread())
|
||
|
{
|
||
|
ScanFolderParams *params = new ScanFolderParams(path, guess, meta, recurse);
|
||
|
if (QueueUserAPC(ScanFolderAPC, scan_thread, (ULONG_PTR)params) == 0)
|
||
|
delete params;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* ---------------
|
||
|
* Scan_ScanFolders
|
||
|
* ---------------
|
||
|
*/
|
||
|
|
||
|
struct ScanFoldersParams
|
||
|
{
|
||
|
ScanFoldersParams(wchar_t **_path, size_t _count, int *_guess, int *_meta, int *_recurse)
|
||
|
{
|
||
|
path = _path;
|
||
|
count = _count;
|
||
|
guess = _guess;
|
||
|
meta = _meta;
|
||
|
recurse = _recurse;
|
||
|
found = 0;
|
||
|
scanned = 0;
|
||
|
cancel_switch = CreateEvent(NULL, TRUE, FALSE, NULL);
|
||
|
status = STATUS_SEARCHING;
|
||
|
ui = 0;
|
||
|
in_timer = 0;
|
||
|
|
||
|
}
|
||
|
~ScanFoldersParams()
|
||
|
{
|
||
|
for (size_t i=0;i!=count;i++)
|
||
|
free(path[i]);
|
||
|
free(path);
|
||
|
free(guess);
|
||
|
free(meta);
|
||
|
free(recurse);
|
||
|
CloseHandle(cancel_switch);
|
||
|
}
|
||
|
wchar_t **path;
|
||
|
size_t count;
|
||
|
int *guess;
|
||
|
int *meta;
|
||
|
int *recurse;
|
||
|
volatile int found;
|
||
|
volatile int scanned;
|
||
|
volatile int status;
|
||
|
int in_timer;
|
||
|
HANDLE cancel_switch;
|
||
|
HWND ui;
|
||
|
};
|
||
|
|
||
|
static VOID CALLBACK ScanFoldersAPC(ULONG_PTR param)
|
||
|
{
|
||
|
// clear extension list to get latest config
|
||
|
if (scan_extlist && scan_extlist != (wchar_t *)1)
|
||
|
GlobalFree((HGLOBAL)scan_extlist);
|
||
|
scan_extlist = 0;
|
||
|
|
||
|
ScanFoldersParams *params = (ScanFoldersParams *)param;
|
||
|
HANDLE events[2] = { scan_killswitch, params->cancel_switch};
|
||
|
|
||
|
for (size_t i=0;i!=params->count;i++)
|
||
|
{
|
||
|
if (ScanCancelled(events, 2))
|
||
|
break;
|
||
|
|
||
|
CountFolder(params->path[i], params->recurse[i], params->cancel_switch, ¶ms->found);
|
||
|
}
|
||
|
|
||
|
params->status = STATUS_GETINFO;
|
||
|
|
||
|
for (size_t i=0;i!=params->count;i++)
|
||
|
{
|
||
|
if (ScanCancelled(events, 2))
|
||
|
break;
|
||
|
|
||
|
int guess = params->guess[i] >= 0 ? params->guess[i] : g_config->ReadInt(L"guessmode",0);;
|
||
|
int meta = params->meta[i] >= 0 ? params->meta[i] : g_config->ReadInt(L"usemetadata",1);
|
||
|
ScanFolder(params->path[i], params->recurse[i], meta, guess, params->cancel_switch, ¶ms->scanned);
|
||
|
}
|
||
|
|
||
|
params->status = STATUS_DONE;
|
||
|
PostMessage(params->ui, WM_APP, 0, 0);
|
||
|
}
|
||
|
|
||
|
static INT_PTR CALLBACK ScanFileUI(HWND hwndDlg, UINT uMsg, WPARAM wParam,LPARAM lParam)
|
||
|
{
|
||
|
switch(uMsg)
|
||
|
{
|
||
|
case WM_INITDIALOG:
|
||
|
{
|
||
|
SetDlgItemTextW(hwndDlg,IDC_STATUS,WASABI_API_LNGSTRINGW(IDS_INITIALIZING));
|
||
|
|
||
|
ScanFoldersParams *params = (ScanFoldersParams *)lParam;
|
||
|
params->ui = hwndDlg;
|
||
|
SetWindowLongPtr(hwndDlg, GWLP_USERDATA, lParam);
|
||
|
if (QueueUserAPC(ScanFoldersAPC, scan_thread, (ULONG_PTR)lParam) == 0)
|
||
|
EndDialog(hwndDlg, 0);
|
||
|
else
|
||
|
{
|
||
|
SendDlgItemMessage(hwndDlg,IDC_PROGRESS1,PBM_SETRANGE,0,MAKELPARAM(0, 100));
|
||
|
SetTimer(hwndDlg,0x123,300,NULL);
|
||
|
}
|
||
|
|
||
|
// show window and restore last position as applicable
|
||
|
POINT pt = {g_config->ReadInt(L"scan_x", -1), g_config->ReadInt(L"scan_y", -1)};
|
||
|
if (!windowOffScreen(hwndDlg, pt))
|
||
|
SetWindowPos(hwndDlg, HWND_TOP, pt.x, pt.y, 0, 0, SWP_NOSIZE | SWP_SHOWWINDOW | SWP_NOSENDCHANGING);
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case WM_TIMER:
|
||
|
{
|
||
|
ScanFoldersParams *params = (ScanFoldersParams *)GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
|
||
|
if (params->in_timer) break;
|
||
|
params->in_timer++;
|
||
|
|
||
|
if(params->status==STATUS_SEARCHING)
|
||
|
{
|
||
|
wchar_t tmp[512] = {0};
|
||
|
StringCchPrintfW(tmp, 512, WASABI_API_LNGSTRINGW(IDS_SEARCHING_X_FILES_FOUND), params->found);
|
||
|
SetDlgItemTextW(hwndDlg,IDC_STATUS,tmp);
|
||
|
}
|
||
|
else if(params->status==STATUS_GETINFO)
|
||
|
{
|
||
|
wchar_t tmp[512] = {0};
|
||
|
int perc=params->found?(params->scanned*100/params->found):0;
|
||
|
StringCchPrintfW(tmp, 512, WASABI_API_LNGSTRINGW(IDS_GETTING_INFO_FROM_FILES_PERCENT),perc);
|
||
|
SetDlgItemTextW(hwndDlg,IDC_STATUS,tmp);
|
||
|
SendDlgItemMessage(hwndDlg,IDC_PROGRESS1,PBM_SETPOS,perc,0);
|
||
|
}
|
||
|
params->in_timer--;
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case WM_APP:
|
||
|
{
|
||
|
KillTimer(hwndDlg,0x123);
|
||
|
SyncTable();
|
||
|
EndDialog(hwndDlg,0);
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case WM_COMMAND:
|
||
|
if (LOWORD(wParam)==IDCANCEL)
|
||
|
{
|
||
|
ScanFoldersParams *params = (ScanFoldersParams *)GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
|
||
|
SetEvent(params->cancel_switch);
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case WM_DESTROY:
|
||
|
{
|
||
|
RECT scan_rect = {0};
|
||
|
GetWindowRect(hwndDlg, &scan_rect);
|
||
|
g_config->WriteInt(L"scan_x", scan_rect.left);
|
||
|
g_config->WriteInt(L"scan_y", scan_rect.top);
|
||
|
|
||
|
KillTimer(hwndDlg,0x123);
|
||
|
ScanFoldersParams *params = (ScanFoldersParams *)GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
|
||
|
SetWindowLongPtr(hwndDlg, GWLP_USERDATA, 0);
|
||
|
delete params;
|
||
|
return FALSE;
|
||
|
}
|
||
|
}
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
/* When you call this function, it will own the memory and release it with free() */
|
||
|
void Scan_ScanFolders(HWND parent, size_t count, wchar_t **paths, int *guess, int *meta, int *recurse)
|
||
|
{
|
||
|
openDb();
|
||
|
|
||
|
if (g_table && ScanCreateThread())
|
||
|
{
|
||
|
ScanFoldersParams *params = new ScanFoldersParams(paths, count, guess, meta, recurse);
|
||
|
WASABI_API_LNG->LDialogBoxParamW(WASABI_API_LNG->FindDllHandleByGUID(WinampLangGUID),
|
||
|
GetModuleHandleW(L"winamp.exe"), IDD_ADDSTUFF,
|
||
|
parent, (DLGPROC)ScanFileUI, (LPARAM)params);
|
||
|
PostMessage(plugin.hwndWinampParent,WM_WA_IPC,NDE_Table_GetRecordsCount(g_table),IPC_STATS_LIBRARY_ITEMCNT);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
for (size_t i=0;i!=count;i++)
|
||
|
free(paths[i]);
|
||
|
free(paths);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void Scan_ScanFolder(HWND parent, const wchar_t *path, int guess, int meta, int recurse)
|
||
|
{
|
||
|
// kind of a hack ...
|
||
|
if (ScanCreateThread())
|
||
|
{
|
||
|
wchar_t **paths = (wchar_t **)calloc(1, sizeof(wchar_t*));
|
||
|
int *guesses = (int *)calloc(1, sizeof(int));
|
||
|
int *metas = (int *)calloc(1, sizeof(int));
|
||
|
int *recs = (int *)calloc(1, sizeof(int));
|
||
|
*guesses = guess;
|
||
|
*metas = meta;
|
||
|
*recs = recurse;
|
||
|
paths[0] = _wcsdup(path);
|
||
|
Scan_ScanFolders(parent, 1, paths, guesses, metas, recs);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static VOID CALLBACK BackgroundScanAPC(ULONG_PTR param)
|
||
|
{
|
||
|
openDb();
|
||
|
if (!g_table)
|
||
|
return;
|
||
|
|
||
|
HANDLE events[2] = { scan_killswitch, scan_cancel};
|
||
|
|
||
|
// clear extension list to get latest config
|
||
|
if (scan_extlist && scan_extlist != (wchar_t *)1)
|
||
|
GlobalFree((HGLOBAL)scan_extlist);
|
||
|
scan_extlist = 0;
|
||
|
|
||
|
// read list from config
|
||
|
UINT codePage = CP_ACP;
|
||
|
char scandirlist[65536] = {0};
|
||
|
if (!g_config->ReadString("scandirlist", 0, scandirlist, 65536))
|
||
|
{
|
||
|
g_config->ReadString("scandirlist_utf8","", scandirlist, 65536);
|
||
|
codePage = CP_UTF8;
|
||
|
}
|
||
|
|
||
|
AutoWide s1(scandirlist, codePage);
|
||
|
size_t len = wcslen(s1)+2;
|
||
|
wchar_t *s =(wchar_t*)calloc(len, sizeof(wchar_t));
|
||
|
if (s)
|
||
|
{
|
||
|
lstrcpynW(s, s1, len);
|
||
|
s[wcslen(s)+1]=0;
|
||
|
|
||
|
wchar_t *p=s;
|
||
|
while (p && *p == L'|') p++;
|
||
|
|
||
|
while ((p=wcsstr(p,L"|")))
|
||
|
{
|
||
|
*p++=0;
|
||
|
while (p && *p == L'|') p++;
|
||
|
}
|
||
|
p=s;
|
||
|
|
||
|
// iterate through list
|
||
|
while (p && *p && !ScanCancelled(events, 2))
|
||
|
{
|
||
|
while (p && *p == L'|') p++;
|
||
|
|
||
|
int use_metadata=g_config->ReadInt(L"usemetadata",1);
|
||
|
int guess_mode=g_config->ReadInt(L"guessmode",0);
|
||
|
int recurse=1;
|
||
|
if (*p == L'<' && wcsstr(p,L">"))
|
||
|
{
|
||
|
p++;
|
||
|
while (p && *p != L'>')
|
||
|
{
|
||
|
// <MmSs>can prefix directory
|
||
|
// M=metadata use override
|
||
|
// m=no metadata
|
||
|
// S=smart guessing
|
||
|
// s=stupid guessing
|
||
|
if (*p == L'M') use_metadata=1;
|
||
|
else if (*p == L'm') use_metadata=0;
|
||
|
else if (*p == L'S') guess_mode=0;
|
||
|
else if (*p == L's') guess_mode=1;
|
||
|
else if (*p == L'r') recurse=0;
|
||
|
else if (*p == L'g') guess_mode=2;
|
||
|
p++;
|
||
|
}
|
||
|
p++;
|
||
|
}
|
||
|
ScanFolder(p, recurse, use_metadata, guess_mode, scan_cancel, 0);
|
||
|
p+=wcslen(p)+1;
|
||
|
}
|
||
|
|
||
|
free(s);
|
||
|
}
|
||
|
|
||
|
/* Remove missing files */
|
||
|
if (!ScanCancelled(events, 2) && g_config->ReadInt(L"bgrescan_compact",1))
|
||
|
{
|
||
|
EnterCriticalSection(&g_db_cs);
|
||
|
nde_scanner_t scanner = NDE_Table_CreateScanner(g_table);
|
||
|
NDE_Scanner_Query(scanner, L"");
|
||
|
NDE_Scanner_First(scanner);
|
||
|
again:
|
||
|
nde_field_t f=NDE_Scanner_GetFieldByID(scanner, MAINTABLE_ID_FILENAME);
|
||
|
wchar_t *gs=0;
|
||
|
if (f)
|
||
|
{
|
||
|
gs = NDE_StringField_GetString(f);
|
||
|
ndestring_retain(gs);
|
||
|
if (GetFileAttributesW(gs) != INVALID_FILE_ATTRIBUTES)
|
||
|
{
|
||
|
NDE_Scanner_Next(scanner);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// Issue wasabi callback for pre removal
|
||
|
WASABI_API_SYSCB->syscb_issueCallback(api_mldb::SYSCALLBACK, api_mldb::MLDB_FILE_REMOVED_PRE, (size_t)gs, 0);
|
||
|
|
||
|
NDE_Scanner_Delete(scanner);
|
||
|
NDE_Scanner_Post(scanner);
|
||
|
|
||
|
// Issue wasabi callback for pre removal
|
||
|
WASABI_API_SYSCB->syscb_issueCallback(api_mldb::SYSCALLBACK, api_mldb::MLDB_FILE_REMOVED_POST, (size_t)gs, 0);
|
||
|
}
|
||
|
}
|
||
|
LeaveCriticalSection(&g_db_cs);
|
||
|
|
||
|
if (f) // done checking for unused files
|
||
|
{
|
||
|
if (IsWindow(g_bgrescan_status_hwnd))
|
||
|
{
|
||
|
wchar_t b[150+MAX_PATH] = {0};
|
||
|
WASABI_API_LNGSTRINGW_BUF(IDS_CHECKING_FOR_FILE, b, 150);
|
||
|
StringCbCatW(b, sizeof(b), PathFindFileNameW(gs));
|
||
|
SetWindowTextW(g_bgrescan_status_hwnd,b);
|
||
|
}
|
||
|
ndestring_release(gs);
|
||
|
gs=0;
|
||
|
if (!ScanCancelled(events, 2))
|
||
|
{
|
||
|
EnterCriticalSection(&g_db_cs);
|
||
|
goto again;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
EnterCriticalSection(&g_db_cs);
|
||
|
NDE_Table_DestroyScanner(g_table, scanner);
|
||
|
LeaveCriticalSection(&g_db_cs);
|
||
|
|
||
|
if (IsWindow(g_bgrescan_status_hwnd))
|
||
|
{
|
||
|
wchar_t b[150] = {0};
|
||
|
WASABI_API_LNGSTRINGW_BUF(IDS_COMPACTING, b, 150);
|
||
|
SetWindowTextW(g_bgrescan_status_hwnd,b);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// TODO: hmmm, safe to do on separate thread?
|
||
|
EnterCriticalSection(&g_db_cs);
|
||
|
if (!ScanCancelled(events, 2))
|
||
|
{
|
||
|
wchar_t *last_query = NULL;
|
||
|
if (m_media_scanner)
|
||
|
{
|
||
|
const wchar_t *lq = NDE_Scanner_GetLastQuery(m_media_scanner);
|
||
|
if (lq) last_query = _wcsdup(lq);
|
||
|
NDE_Table_DestroyScanner(g_table, m_media_scanner);
|
||
|
}
|
||
|
NDE_Table_Sync(g_table); // this is currently b0rk3d -- fucko :)
|
||
|
NDE_Table_Compact(g_table);
|
||
|
g_table_dirty=0;
|
||
|
if (m_media_scanner)
|
||
|
{
|
||
|
m_media_scanner=NDE_Table_CreateScanner(g_table);
|
||
|
if (last_query != NULL)
|
||
|
{
|
||
|
NDE_Scanner_Query(m_media_scanner, last_query);
|
||
|
free(last_query);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
PostMessage(plugin.hwndWinampParent,WM_WA_IPC,NDE_Table_GetRecordsCount(g_table),IPC_STATS_LIBRARY_ITEMCNT);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
NDE_Table_Sync(g_table); // this is currently b0rk3d -- fucko :)
|
||
|
g_table_dirty=0;
|
||
|
}
|
||
|
LeaveCriticalSection(&g_db_cs);
|
||
|
if (IsWindow(g_bgrescan_status_hwnd))
|
||
|
SetWindowTextW(g_bgrescan_status_hwnd,L"");
|
||
|
g_bgscan_last_rescan = time(NULL);
|
||
|
g_bgscan_scanning = 0;
|
||
|
}
|
||
|
|
||
|
void Scan_BackgroundScan()
|
||
|
{
|
||
|
if (ScanCreateThread())
|
||
|
{
|
||
|
g_bgrescan_force = 0;
|
||
|
g_bgscan_last_rescan = time(NULL);
|
||
|
g_bgscan_scanning = 1;
|
||
|
|
||
|
QueueUserAPC(BackgroundScanAPC, scan_thread, (ULONG_PTR)0);
|
||
|
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void RemoveFiles(HANDLE cancelswitch, volatile int *found, volatile int *count, volatile int *scanned)
|
||
|
{
|
||
|
// TODO: benski> we might need to keep the database lock the whole time. need to think it thru
|
||
|
EnterCriticalSection(&g_db_cs);
|
||
|
nde_scanner_t myscanner=NDE_Table_CreateScanner(g_table);
|
||
|
NDE_Scanner_Query(myscanner, L"");
|
||
|
NDE_Scanner_First(myscanner);
|
||
|
*found=0;
|
||
|
*scanned=0;
|
||
|
*count=NDE_Table_GetRecordsCount(g_table);
|
||
|
LeaveCriticalSection(&g_db_cs);
|
||
|
|
||
|
HANDLE events[2] = { scan_killswitch, cancelswitch};
|
||
|
|
||
|
bool fileRemoved = false;
|
||
|
wchar_t *filename;
|
||
|
while (!ScanCancelled(events, 2))
|
||
|
{
|
||
|
EnterCriticalSection(&g_db_cs);
|
||
|
if (!NDE_Scanner_BOF(myscanner) && !NDE_Scanner_EOF(myscanner))
|
||
|
{
|
||
|
nde_field_t f= NDE_Scanner_GetFieldByID(myscanner, MAINTABLE_ID_FILENAME);
|
||
|
if (f)
|
||
|
{
|
||
|
(*scanned)++;
|
||
|
|
||
|
filename = (NDE_StringField_GetString(f));
|
||
|
|
||
|
if (GetFileAttributesW(NDE_StringField_GetString(f)) != INVALID_FILE_ATTRIBUTES)
|
||
|
{
|
||
|
NDE_Scanner_Next(myscanner);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// Issue wasabi callback for pre removal
|
||
|
WASABI_API_SYSCB->syscb_issueCallback(api_mldb::SYSCALLBACK, api_mldb::MLDB_FILE_REMOVED_PRE, (size_t)filename, 0);
|
||
|
|
||
|
//remove file
|
||
|
NDE_Scanner_Delete(myscanner);
|
||
|
NDE_Scanner_Post(myscanner);
|
||
|
(*found)++;
|
||
|
fileRemoved = true;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
//remove file
|
||
|
NDE_Scanner_Delete(myscanner);
|
||
|
NDE_Scanner_Post(myscanner);
|
||
|
(*found)++;
|
||
|
fileRemoved = false;
|
||
|
}
|
||
|
}
|
||
|
else // last file
|
||
|
{
|
||
|
LeaveCriticalSection(&g_db_cs);
|
||
|
break;
|
||
|
}
|
||
|
LeaveCriticalSection(&g_db_cs);
|
||
|
|
||
|
if (fileRemoved)
|
||
|
{
|
||
|
// Issue wasabi callback for pre removal
|
||
|
WASABI_API_SYSCB->syscb_issueCallback(api_mldb::SYSCALLBACK, api_mldb::MLDB_FILE_REMOVED_POST, (size_t)filename, 0);
|
||
|
fileRemoved = false;
|
||
|
}
|
||
|
}
|
||
|
EnterCriticalSection(&g_db_cs);
|
||
|
NDE_Table_DestroyScanner(g_table, myscanner); // important that we delete the scanner BEFORE
|
||
|
myscanner=0;
|
||
|
|
||
|
wchar_t *last_query = NULL;
|
||
|
if (m_media_scanner)
|
||
|
{
|
||
|
const wchar_t *lq = NDE_Scanner_GetLastQuery(m_media_scanner);
|
||
|
if (lq) last_query = _wcsdup(lq);
|
||
|
NDE_Table_DestroyScanner(g_table, m_media_scanner);
|
||
|
}
|
||
|
NDE_Table_Sync(g_table); // this is currently b0rk3d -- fucko :)
|
||
|
NDE_Table_Compact(g_table);
|
||
|
g_table_dirty=0;
|
||
|
if (m_media_scanner)
|
||
|
{
|
||
|
m_media_scanner=NDE_Table_CreateScanner(g_table);
|
||
|
if (last_query != NULL)
|
||
|
{
|
||
|
NDE_Scanner_Query(m_media_scanner, last_query);
|
||
|
free(last_query);
|
||
|
}
|
||
|
}
|
||
|
LeaveCriticalSection(&g_db_cs);
|
||
|
}
|
||
|
|
||
|
struct RemoveFilesParams
|
||
|
{
|
||
|
RemoveFilesParams()
|
||
|
{
|
||
|
found = 0;
|
||
|
scanned = 0;
|
||
|
total = 0;
|
||
|
cancel_switch = CreateEvent(NULL, TRUE, FALSE, NULL);
|
||
|
ui = 0;
|
||
|
in_timer = 0;
|
||
|
}
|
||
|
~RemoveFilesParams()
|
||
|
{
|
||
|
|
||
|
CloseHandle(cancel_switch);
|
||
|
}
|
||
|
volatile int found;
|
||
|
volatile int scanned;
|
||
|
volatile int total;
|
||
|
|
||
|
int in_timer;
|
||
|
HANDLE cancel_switch;
|
||
|
HWND ui;
|
||
|
};
|
||
|
|
||
|
static VOID CALLBACK RemoveFilesAPC(ULONG_PTR param)
|
||
|
{
|
||
|
RemoveFilesParams *params = (RemoveFilesParams *)param;
|
||
|
|
||
|
RemoveFiles(params->cancel_switch, ¶ms->found, ¶ms->total, ¶ms->scanned);
|
||
|
PostMessage(params->ui, WM_APP, 0, 0);
|
||
|
}
|
||
|
|
||
|
static INT_PTR CALLBACK RemoveFilesUI(HWND hwndDlg, UINT uMsg, WPARAM wParam,LPARAM lParam)
|
||
|
{
|
||
|
switch(uMsg)
|
||
|
{
|
||
|
case WM_INITDIALOG:
|
||
|
{
|
||
|
SetWindowTextW(hwndDlg,WASABI_API_LNGSTRINGW(IDS_REMOVING_FILES_NOT_EXISTING));
|
||
|
SetDlgItemTextW(hwndDlg,IDC_STATUS,WASABI_API_LNGSTRINGW(IDS_INITIALIZING));
|
||
|
|
||
|
RemoveFilesParams *params = (RemoveFilesParams *)lParam;
|
||
|
params->ui = hwndDlg;
|
||
|
SetWindowLongPtr(hwndDlg, GWLP_USERDATA, lParam);
|
||
|
if (QueueUserAPC(RemoveFilesAPC, scan_thread, (ULONG_PTR)lParam) == 0)
|
||
|
EndDialog(hwndDlg, 0);
|
||
|
else
|
||
|
{
|
||
|
SendDlgItemMessage(hwndDlg,IDC_PROGRESS1,PBM_SETRANGE,0,MAKELPARAM(0, 100));
|
||
|
SetTimer(hwndDlg,0x123,300,NULL);
|
||
|
}
|
||
|
|
||
|
// show window and restore last position as applicable
|
||
|
POINT pt = {g_config->ReadInt(L"scan_x", -1), g_config->ReadInt(L"scan_y", -1)};
|
||
|
if (!windowOffScreen(hwndDlg, pt))
|
||
|
SetWindowPos(hwndDlg, HWND_TOP, pt.x, pt.y, 0, 0, SWP_NOSIZE | SWP_SHOWWINDOW | SWP_NOSENDCHANGING);
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case WM_TIMER:
|
||
|
{
|
||
|
RemoveFilesParams *params = (RemoveFilesParams *)GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
|
||
|
if (params->in_timer) break;
|
||
|
params->in_timer++;
|
||
|
|
||
|
if(params->total)
|
||
|
{
|
||
|
wchar_t tmp[512] = {0};
|
||
|
int perc=(params->scanned*100/(params->total?params->total:1));
|
||
|
StringCchPrintfW(tmp, 512, WASABI_API_LNGSTRINGW(IDS_SCANNING_X_OF_X_X_REMOVED),params->scanned,params->total,params->found);
|
||
|
SetDlgItemTextW(hwndDlg,IDC_STATUS,tmp);
|
||
|
SendDlgItemMessage(hwndDlg,IDC_PROGRESS1,PBM_SETPOS,perc,0);
|
||
|
}
|
||
|
|
||
|
params->in_timer--;
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case WM_APP:
|
||
|
{
|
||
|
RemoveFilesParams *params = (RemoveFilesParams *)GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
|
||
|
KillTimer(hwndDlg,0x123);
|
||
|
|
||
|
wchar_t tmp[512] = {0};
|
||
|
int perc=(params->scanned*100/(params->total?params->total:1));
|
||
|
StringCchPrintfW(tmp, 512, WASABI_API_LNGSTRINGW(IDS_SCANNED_X_FILES_X_REMOVED),params->total,params->found);
|
||
|
SetDlgItemTextW(hwndDlg,IDC_STATUS,tmp);
|
||
|
SendDlgItemMessage(hwndDlg,IDC_PROGRESS1,PBM_SETPOS,perc,0);
|
||
|
|
||
|
SyncTable();
|
||
|
EndDialog(hwndDlg,0);
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case WM_COMMAND:
|
||
|
if (LOWORD(wParam)==IDCANCEL)
|
||
|
{
|
||
|
RemoveFilesParams *params = (RemoveFilesParams *)GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
|
||
|
SetEvent(params->cancel_switch);
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case WM_DESTROY:
|
||
|
{
|
||
|
RECT scan_rect = {0};
|
||
|
GetWindowRect(hwndDlg, &scan_rect);
|
||
|
g_config->WriteInt(L"scan_x", scan_rect.left);
|
||
|
g_config->WriteInt(L"scan_y", scan_rect.top);
|
||
|
|
||
|
KillTimer(hwndDlg,0x123);
|
||
|
RemoveFilesParams *params = (RemoveFilesParams *)GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
|
||
|
SetWindowLongPtr(hwndDlg, GWLP_USERDATA, 0);
|
||
|
delete params;
|
||
|
}
|
||
|
return FALSE;
|
||
|
}
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
void Scan_RemoveFiles(HWND parent)
|
||
|
{
|
||
|
openDb();
|
||
|
if (g_table && ScanCreateThread())
|
||
|
{
|
||
|
RemoveFilesParams *params = new RemoveFilesParams();
|
||
|
WASABI_API_LNG->LDialogBoxParamW(WASABI_API_LNG->FindDllHandleByGUID(WinampLangGUID),
|
||
|
GetModuleHandleW(L"winamp.exe"), IDD_ADDSTUFF,
|
||
|
parent, (DLGPROC)RemoveFilesUI, (LPARAM)params);
|
||
|
PostMessage(plugin.hwndWinampParent,WM_WA_IPC,NDE_Table_GetRecordsCount(g_table),IPC_STATS_LIBRARY_ITEMCNT);
|
||
|
}
|
||
|
}
|