保存文件的Windows事件挂钩

时间:2016-11-30 17:02:52

标签: c++ windows events

每次我保存我在NotePad ++中处理的某个脚本文件时,我都需要将更改上传到我们的服务器,以便我们可以将更改部署到各种计算机上。

我有时忘记在NotePad ++中重构我的代码后上传更改,我想知道是否有一种方法可以创建一个简单的应用程序来监听“保存”事件并自动为我上传文件。 / p>

我目前正在Windows操作系统上运行,并希望使用C ++来实现这一点。我想探索Windows事件,并可能绑定到事件挂钩来实现这一目标。任何其他语言也会受到欢迎。

任何想法或提示?

到目前为止,我的代码遵循以下Josh的建议:

#include <windows.h>
#include <stdlib.h>
#include <stdio.h>
#include <tchar.h>

void RefreshDirectory(LPTSTR);
void WatchDirectory(LPTSTR);

void _tmain(int argc, TCHAR *argv[])
{
    if (argc != 2)
    {
        _tprintf(TEXT("Usage: %s <dir>\n"), argv[0]);
        return;
    }

    WatchDirectory(argv[1]);
}

void WatchDirectory(LPTSTR lpDir)
{
    DWORD dwWaitStatus;
    HANDLE dwChangeHandles[2];
    TCHAR lpDrive[4];
    TCHAR lpFile[_MAX_FNAME];
    TCHAR lpExt[_MAX_EXT];

    _tsplitpath_s(lpDir, lpDrive, 4, NULL, 0, lpFile, _MAX_FNAME, lpExt, _MAX_EXT);

    lpDrive[2] = (TCHAR)'\\';
    lpDrive[3] = (TCHAR)'\0';

    // Watch the directory for file creation and deletion. 

    dwChangeHandles[0] = FindFirstChangeNotification(
        lpDir,                         // directory to watch 
        FALSE,                         // do not watch subtree 
        FILE_NOTIFY_CHANGE_LAST_WRITE); // watch file name changes 

    if (dwChangeHandles[0] == INVALID_HANDLE_VALUE)
    {
        printf("\n ERROR: FindFirstChangeNotification function failed.\n");
        ExitProcess(GetLastError());
    }

    // Make a final validation check on our handles.

    if ((dwChangeHandles[0] == NULL))
    {
        printf("\n ERROR: Unexpected NULL from FindFirstChangeNotification.\n");
        ExitProcess(GetLastError());
    }

    // Change notification is set. Now wait on both notification 
    // handles and refresh accordingly. 

    while (TRUE)
    {
        // Wait for notification.

        printf("\nWaiting for notification...\n");

        // Waits until the specified object is in the signaled state or 
        // the time-out interval elapses.
        // Because our second parameter is set to INFINITE, the function will
        // return only when the object is signaled.
        dwWaitStatus = WaitForSingleObject(dwChangeHandles, INFINITE);

        switch (dwWaitStatus)
        {
            // Our return value, WAIT_OBJECT_0 signifies that the first object
            // signaled the event.
            case WAIT_OBJECT_0:

                // A file was created, renamed, or deleted in the directory.
                // Refresh this directory and restart the notification.

                RefreshDirectory(lpDir);
                if (FindNextChangeNotification(dwChangeHandles[0]) == FALSE)
                {
                    printf("\n ERROR: FindNextChangeNotification function failed.\n");
                    ExitProcess(GetLastError());
                }
                break;

            case WAIT_TIMEOUT:

                // A timeout occurred, this would happen if some value other 
                // than INFINITE is used in the Wait call and no changes occur.
                // In a single-threaded environment you might not want an
                // INFINITE wait.

                printf("\nNo changes in the timeout period.\n");
                break;

            default:
                printf("\n ERROR: Unhandled dwWaitStatus.\n");
                ExitProcess(GetLastError());
                break;
        }
    }
}

void RefreshDirectory(LPTSTR lpDir)
{
    // This is where you might place code to refresh your
    // directory listing, but not the subtree because it
    // would not be necessary.

    _tprintf(TEXT("Directory (%s) changed.\n"), lpDir);
}

2 个答案:

答案 0 :(得分:1)

您可以使用FindFirstChangeNotification监控文件系统的更改。当你调用这个函数时,你会得到一个HANDLE。您可以使用WaitSingleObject(或类似)等待该句柄。当等待返回时,您可以使用ReadDirectoryChanges来确切地知道发生了什么。如果发生的任何事件与某些事件相匹配或更改您关心的文件,您可以采取适当的行动......否则忽略该事件。

因为您正在等待(从而阻止该线程),如果您希望您的程序正在执行其他任何操作,您可能希望在工作线程上执行此工作。

一种简单的启动方式可能是使用FILE_NOTIFY_CHANGE_LAST_WRITE过滤器监听事件;当写入受监视目录中的文件时,这将释放您的等待。

请注意,并非所有程序都以相同的方式保存文件;一些打开现有文件并写入它,其他人删除它并替换,或其某些组合(首先写入临时文件,然后将其与原始文件交换)。因此,它可能不像等待只是最后写入通知那样直接完成您之后所做的事情。

答案 1 :(得分:1)

考虑改为编写NP ++插件。

您可以注册插件,以便在使用NPPN_FILESAVEDNPPN_FILEBEFORESAVE保存或即将保存文件时收到通知。

请看这个链接:

based off this answer