Internal new background directory scanning for Windows; see issue #597
This commit is contained in:
@@ -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)
|
||||||
|
@@ -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) {
|
||||||
|
@@ -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) {
|
||||||
|
@@ -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
|
||||||
|
|
||||||
|
@@ -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
|
||||||
|
@@ -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;
|
||||||
|
@@ -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);
|
||||||
|
@@ -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:
|
||||||
|
Reference in New Issue
Block a user