Internal new background directory scanning for Windows; see issue #597

This commit is contained in:
Oliver Duis
2011-03-25 22:17:16 +01:00
parent 0f416b379a
commit adc5518d52
8 changed files with 32 additions and 33 deletions

View File

@@ -518,7 +518,7 @@ Thumbnail::Thumbnail () :
Thumbnail::~Thumbnail () { Thumbnail::~Thumbnail () {
delete thumbImg; delete thumbImg; thumbImg=NULL;
delete [] aeHistogram; delete [] aeHistogram;
delete [] embProfileData; delete [] embProfileData;
if (embProfile) if (embProfile)

View File

@@ -140,6 +140,8 @@ void DirBrowser::updateVolumes () {
int nvolumes = GetLogicalDrives (); int nvolumes = GetLogicalDrives ();
if (nvolumes!=volumes) { if (nvolumes!=volumes) {
gdk_threads_enter();
for (int i=0; i<32; i++) for (int i=0; i<32; i++)
if (((volumes >> i) & 1) && !((nvolumes >> i) & 1)) { // volume i has been deleted if (((volumes >> i) & 1) && !((nvolumes >> i) & 1)) { // volume i has been deleted
for (Gtk::TreeModel::iterator iter = dirTreeModel->children().begin(); iter!=dirTreeModel->children().end(); iter++) for (Gtk::TreeModel::iterator iter = dirTreeModel->children().begin(); iter!=dirTreeModel->children().end(); iter++)
@@ -151,14 +153,14 @@ void DirBrowser::updateVolumes () {
else if (!((volumes >> i) & 1) && ((nvolumes >> i) & 1)) else if (!((volumes >> i) & 1) && ((nvolumes >> i) & 1))
addRoot ('A'+i); // volume i has been added addRoot ('A'+i); // volume i has been added
volumes = nvolumes; volumes = nvolumes;
gdk_threads_leave();
} }
} }
int _updateVolumes (void* br) { int _updateVolumes (void* br) {
gdk_threads_enter ();
((DirBrowser*)br)->updateVolumes (); ((DirBrowser*)br)->updateVolumes ();
gdk_threads_leave ();
return 1; return 1;
} }
int _updateDirTree (void* br) { int _updateDirTree (void* br) {

View File

@@ -190,7 +190,7 @@ struct addparams {
FileBrowserEntry* entry; FileBrowserEntry* entry;
}; };
int addfl (void* data) { int AddEntryUIThread (void* data) {
addparams* ap = (addparams*) data; addparams* ap = (addparams*) data;
FileBrowserIdleHelper* fbih = ap->fbih; FileBrowserIdleHelper* fbih = ap->fbih;
@@ -222,7 +222,7 @@ void FileBrowser::addEntry (FileBrowserEntry* entry) {
addparams* ap = new addparams; addparams* ap = new addparams;
ap->fbih = fbih; ap->fbih = fbih;
ap->entry = entry; ap->entry = entry;
g_idle_add (addfl, ap); g_idle_add (AddEntryUIThread, ap);
} }
void FileBrowser::addEntry_ (FileBrowserEntry* entry) { void FileBrowser::addEntry_ (FileBrowserEntry* entry) {

View File

@@ -36,19 +36,6 @@
extern Glib::ustring argv0; extern Glib::ustring argv0;
#ifdef _WIN32
int _directoryUpdater (void* cat) {
((FileCatalog*)cat)->checkCounter++;
if (((FileCatalog*)cat)->checkCounter==2) {
gdk_threads_enter ();
((FileCatalog*)cat)->reparseDirectory ();
gdk_threads_leave ();
}
return 1;
}
#endif
FileCatalog::FileCatalog (CoarsePanel* cp, ToolBar* tb, FilePanel* filepanel) : FileCatalog::FileCatalog (CoarsePanel* cp, ToolBar* tb, FilePanel* filepanel) :
selectedDirectoryId(1), selectedDirectoryId(1),
listener(NULL), listener(NULL),
@@ -235,8 +222,6 @@ FileCatalog::FileCatalog (CoarsePanel* cp, ToolBar* tb, FilePanel* filepanel) :
selectedDirectory = ""; selectedDirectory = "";
#ifdef _WIN32 #ifdef _WIN32
wdMonitor = NULL; wdMonitor = NULL;
checkCounter = 2;
g_timeout_add (CHECKTIME, _directoryUpdater, this);
#endif #endif
} }
@@ -958,9 +943,16 @@ int FileCatalog::reparseDirectory () {
} }
#ifdef _WIN32 #ifdef _WIN32
void FileCatalog::winDirChanged () { int winDirChangedUITread (void* cat) {
gdk_threads_enter ();
((FileCatalog*)cat)->reparseDirectory ();
gdk_threads_leave ();
checkCounter = 0; return 0;
}
void FileCatalog::winDirChanged () {
g_idle_add(winDirChangedUITread, this);
} }
#endif #endif

View File

@@ -115,7 +115,6 @@ class FileCatalog : public Gtk::VBox,
#ifdef _WIN32 #ifdef _WIN32
WinDirMonitor* wdMonitor; WinDirMonitor* wdMonitor;
public: public:
int checkCounter;
void winDirChanged (); void winDirChanged ();
private: private:
#endif #endif

View File

@@ -140,6 +140,7 @@ bool Thumbnail::isSupported () {
} }
const ProcParams& Thumbnail::getProcParams () { const ProcParams& Thumbnail::getProcParams () {
Glib::Mutex::Lock lock(mutex);
if (pparamsValid) if (pparamsValid)
return pparams; return pparams;
@@ -160,6 +161,7 @@ const ProcParams& Thumbnail::getProcParams () {
} }
void Thumbnail::loadProcParams () { void Thumbnail::loadProcParams () {
Glib::Mutex::Lock lock(mutex);
pparamsValid = false; pparamsValid = false;
if (options.paramsLoadLocation==PLL_Input) { if (options.paramsLoadLocation==PLL_Input) {
@@ -181,6 +183,7 @@ void Thumbnail::loadProcParams () {
} }
void Thumbnail::clearProcParams (int whoClearedIt) { void Thumbnail::clearProcParams (int whoClearedIt) {
Glib::Mutex::Lock lock(mutex);
cfs.recentlySaved = false; cfs.recentlySaved = false;
pparamsValid = false; pparamsValid = false;
@@ -208,6 +211,7 @@ bool Thumbnail::hasProcParams () {
} }
void Thumbnail::setProcParams (const ProcParams& pp, int whoChangedIt, bool updateCacheNow) { void Thumbnail::setProcParams (const ProcParams& pp, int whoChangedIt, bool updateCacheNow) {
Glib::Mutex::Lock lock(mutex);
if (pparams!=pp) if (pparams!=pp)
cfs.recentlySaved = false; cfs.recentlySaved = false;
@@ -514,6 +518,7 @@ void Thumbnail::updateCache () {
} }
Thumbnail::~Thumbnail () { Thumbnail::~Thumbnail () {
Glib::Mutex::Lock lock(mutex);
delete [] lastImg; delete [] lastImg;
delete tpp; delete tpp;

View File

@@ -19,17 +19,19 @@
#include <windirmonitor.h> #include <windirmonitor.h>
static void CALLBACK current_directory_monitor_callback (DWORD error, DWORD nBytes, LPOVERLAPPED lpOverlapped) { static void CALLBACK current_directory_monitor_callback (DWORD error, DWORD nBytes, LPOVERLAPPED lpOverlapped) {
// printf ("hivas: %d %d %d\n", error, nBytes, lpOverlapped);
WinDirMonitor::MonitorData* monData = (WinDirMonitor::MonitorData*)lpOverlapped; WinDirMonitor::MonitorData* monData = (WinDirMonitor::MonitorData*)lpOverlapped;
if (!nBytes) { if (!nBytes) {
delete monData; delete monData;
return; return;
} }
if (monData->listener) // ReadDirectoryChangesW sometimes emits multiple events per change (one for each change type)
// To make sure it's not flooding update, this gets filtered.
time_t curTime= ::time(NULL);
if (monData->listener && ::difftime(curTime, monData->lastTimeUpdateDir)>1.0) {
monData->listener->winDirChanged (); monData->listener->winDirChanged ();
monData->lastTimeUpdateDir = curTime;
}
ReadDirectoryChangesW (monData->hDirectory, ReadDirectoryChangesW (monData->hDirectory,
monData->file_notify_buffer, monData->file_notify_buffer,
@@ -37,16 +39,14 @@ static void CALLBACK current_directory_monitor_callback (DWORD error, DWORD nByt
FALSE, FALSE,
FILE_NOTIFY_CHANGE_FILE_NAME | FILE_NOTIFY_CHANGE_FILE_NAME |
FILE_NOTIFY_CHANGE_DIR_NAME | FILE_NOTIFY_CHANGE_DIR_NAME |
FILE_NOTIFY_CHANGE_ATTRIBUTES | FILE_NOTIFY_CHANGE_LAST_WRITE,
FILE_NOTIFY_CHANGE_SIZE,
&monData->buffer_filled_bytes, &monData->buffer_filled_bytes,
&monData->overlapped, &monData->overlapped,
current_directory_monitor_callback); current_directory_monitor_callback);
} }
WinDirMonitor::WinDirMonitor (Glib::ustring dirName, WinDirChangeListener* listener) : monData(NULL) { WinDirMonitor::WinDirMonitor (Glib::ustring dirName, WinDirChangeListener* listener) : monData(NULL) {
wchar_t* wdirname = (wchar_t*)g_utf8_to_utf16 (dirName.c_str(), -1, NULL, NULL, NULL);
wchar_t* wdirname = (wchar_t*)g_utf8_to_utf16 (dirName.raw().c_str(), -1, NULL, NULL, NULL);
HANDLE hDirectory = CreateFileW (wdirname, FILE_LIST_DIRECTORY,FILE_SHARE_DELETE | FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS | FILE_FLAG_OVERLAPPED, NULL); HANDLE hDirectory = CreateFileW (wdirname, FILE_LIST_DIRECTORY,FILE_SHARE_DELETE | FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS | FILE_FLAG_OVERLAPPED, NULL);
g_free (wdirname); g_free (wdirname);
@@ -57,6 +57,7 @@ WinDirMonitor::WinDirMonitor (Glib::ustring dirName, WinDirChangeListener* liste
monData->buffer_allocated_bytes = 32768; monData->buffer_allocated_bytes = 32768;
monData->file_notify_buffer = new char [monData->buffer_allocated_bytes]; monData->file_notify_buffer = new char [monData->buffer_allocated_bytes];
monData->hDirectory = hDirectory; monData->hDirectory = hDirectory;
monData->lastTimeUpdateDir = ::time(NULL);
// printf ("mondata=%d\n", monData); // printf ("mondata=%d\n", monData);
ReadDirectoryChangesW (monData->hDirectory, ReadDirectoryChangesW (monData->hDirectory,
@@ -65,8 +66,7 @@ WinDirMonitor::WinDirMonitor (Glib::ustring dirName, WinDirChangeListener* liste
FALSE, FALSE,
FILE_NOTIFY_CHANGE_FILE_NAME | FILE_NOTIFY_CHANGE_FILE_NAME |
FILE_NOTIFY_CHANGE_DIR_NAME | FILE_NOTIFY_CHANGE_DIR_NAME |
FILE_NOTIFY_CHANGE_ATTRIBUTES | FILE_NOTIFY_CHANGE_LAST_WRITE,
FILE_NOTIFY_CHANGE_SIZE,
&monData->buffer_filled_bytes, &monData->buffer_filled_bytes,
&monData->overlapped, &monData->overlapped,
current_directory_monitor_callback); current_directory_monitor_callback);

View File

@@ -39,6 +39,7 @@ class WinDirMonitor : public Glib::Object {
HANDLE hDirectory; HANDLE hDirectory;
WinDirChangeListener* listener; WinDirChangeListener* listener;
int bigyo; int bigyo;
time_t lastTimeUpdateDir; // for filtering multiple updates events
}; };
private: private: