实现托管属性处理程序Shell扩展的正确方法是什么?

Dav*_*nch 5 .net windows shell32 com-interop

既然.NET CLR 4.0支持并行(SxS)操作,现在应该可以在托管代码中编写shell扩展.我试过这个并成功编写了一个实现IPropertyStore,IInitializeWithStream和IPropertyStoreCapabilities的Property Handler.

处理程序工作正常,并在通过资源管理器浏览文件时按预期调用.它还可以在预览面板和文件属性"详细信息"面板中显示自定义属性.

但是,当我尝试在预览面板中编辑属性,然后单击"保存"时,出现"正在使用文件"错误,指出该文件在Windows资源管理器中已打开.

一些花絮:

  1. 当资源管理器调用IInitializeWithStream.Initialize时,STGM属性设置为STGM_SHARE_DENY_WRITE.
  2. 并且探测器决不会调用IPropertyStore.SetValue或IPropertyStore.Commit.
  3. 我看到在不同线程上对我的处理程序重复调用相同的文件属性.

那么我需要更改(或设置在注册表中)以使属性保存工作?

更新:

感谢Ben,我已经开始工作了."困难部分"(至少对我来说)是理解COM互操作永远不会在我的PropertyHandler上调用Dispose或Finalize.这使我处理的文件保持打开状态,直到GC运行.

幸运的是,"属性处理程序协议"的作用是,当为ReadValue()调用IInitializeWithSream.Initialize()时,streamMode是ReadOnly,当为SetValue()调用它时,streamMode是ReadWrite,并且将调用Commit()在末尾.

int IInitializeWithStream.Initialize( IStream stream, uint grfMode )
{
    _stream = stream;
    _streamMode = (Stgm)grfMode;

    Load();

    // We release here cause if this is a read operation we won't get called back, 
    // and our finializer isn't called. 
    if ( ( _streamMode & Stgm.ReadWrite ) != Stgm.ReadWrite )
    {
        Marshal.ReleaseComObject( _stream );
        _stream = null;
    }
    return HResult.S_OK;
}

int IPropertyStore.Commit()
{
    bool result = false;

    if ( _stream != null )
    {
        result = WriteStream( _stream );
        Marshal.ReleaseComObject( _stream );
        _stream = null;
    }

    return result ? HResult.S_OK : HResult.E_FAIL;
}
Run Code Online (Sandbox Code Playgroud)

小智 3

是的,您必须 AddRef() 流以使其保持打开状态并保持引用正确活动。

请注意,索引器也将使用您的属性处理程序来打开文件。因此,如果泄漏流对象,文件将保持打开状态。您可以使用 sysinternals procexp 来告诉哪个进程打开了文件,或者使用 procmon 来告诉它使用了哪些调用和参数。