Как обнаружить изменения в перенаправлении известных папок

Когда OneDrive установлен с Перемещение известных папок включено, первый вход пользователя начинается без перенаправления известных папок, затем в середине сеанса они перенаправляются в папки OneDrive.

Как я могу это обнаружить?

Я уже пробовал WM_SETTINGCHANGE и SHChangeNotifyRegister на CSIDL_Desktop безрезультатно.


person Mitch    schedule 28.08.2020    source источник


Ответы (1)


Это грязный недокументированный взлом, но если вы отслеживаете HKEY_CURRENT_USER\Software\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders на наличие изменения, он сработает при перенаправлении.

Пример С# того, как это сделать:

public sealed class KnownFolderWatcher : IDisposable
{
    private readonly SynchronizationContext SyncCtx;
    private readonly RegistryKey Key;
    private readonly Thread thRead;
    private readonly AutoResetEvent mreTriggered;

    public event EventHandler KeyChanged;

    private Exception Exception;
    private bool isDisposed;


    public KnownFolderWatcher(SynchronizationContext syncCtx)
    {
        this.SyncCtx = syncCtx ?? throw new ArgumentNullException(nameof(syncCtx));

        this.Key = Registry.CurrentUser.OpenSubKey(@"Software\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders", false)
            ?? throw new InvalidOperationException("Could not open User Shell Folders key");

        this.mreTriggered = new AutoResetEvent(false);

        this.thRead = new Thread(thRead_Main);
        this.thRead.Name = "Registry Change Reader";
        this.thRead.IsBackground = true;
        this.thRead.Start();
    }

    private void thRead_Main()
    {
        try
        {
            while (true)
            {
                NativeMethods.RegNotifyChangeKeyValue(Key.Handle, false, 4 /* REG_NOTIFY_CHANGE_LAST_SET */, mreTriggered.SafeWaitHandle, true);
                mreTriggered.WaitOne();
                if (isDisposed)
                {
                    break;
                }

                SyncCtx.Post(_1 =>
                {
                    KeyChanged?.Invoke(this, EventArgs.Empty);
                }, null);
            }
        }
        catch (Exception ex)
        {
            this.Exception = ex;
        }
    }

    public void Dispose()
    {
        if (isDisposed)
        {
            throw new ObjectDisposedException(nameof(KnownFolderWatcher));
        }
        isDisposed = true;

        mreTriggered.Set();
        thRead.Join();

        if (this.Exception != null)
        {
            throw new InvalidOperationException("Exception from read thread", Exception);
        }
    }
}

internal static class NativeMethods
{
    [DllImport("advapi32.dll", ExactSpelling = true, SetLastError = true)]
    internal static extern uint RegNotifyChangeKeyValue(SafeRegistryHandle key, bool watchSubTree, uint notifyFilter, SafeWaitHandle regEvent, bool async);
}
person Mitch    schedule 28.08.2020