0

我正在使用 python2.7 和 py2exe 从我的 python 脚本创建 dll。

成功创建了一个 DLL 并为我的图标覆盖状态注册了一个条目,然后通过任务管理器重新启动了 Windows 资源管理器进程。

验证我的条目是否在注册表中,是的,它在那里。

但是,当我通过 python 测试应用程序脚本为特定文件夹位置设置我的状态时。

我希望所选路径中的所有文件和文件夹都应该覆盖我的覆盖图标。

但是不,图标覆盖根本没有发生。

但是当我通过python脚本测试注册条目(不创建DLL)并通过我的测试应用程序脚本设置我的图标覆盖时。

是的,它工作得很好。

我很困惑为什么在尝试使用我的 DLL 时它没有发生???

下面是我注册状态条目的python脚本

import os
import win32traceutil
import pythoncom
import winerror
from win32com.shell import shell, shellcon
from multiprocessing.connection import Client

REG_PATH = r'Software\Microsoft\Windows\CurrentVersion\Explorer\ShellIconOverlayIdentifiers'
REG_KEY  = "IconOverlayTest"

class IconOverlay:
    _reg_desc_       = 'Icon Overlay COM Server'
    _public_methods_ = ['GetOverlayInfo', 'GetPriority','IsMemberOf']
    _com_interfaces_ = [shell.IID_IShellIconOverlayIdentifier, pythoncom.IID_IDispatch]
    def __init__(self, *_args, **_kwargs):
        self._icon = None
        self._icon_id = None
        raise NotImplementedError
    def GetOverlayInfo(self):
        return self._icon, 0, shellcon.ISIOI_ICONFILE
    def GetPriority(self):
        return 0
    def IsMemberOf(self, path, _attrs):
        if is_member(path, self._icon_id):
            return winerror.S_OK
        return winerror.E_FAIL

class IconOverlay_test(IconOverlay):
    _reg_progid_     = 'a.TestServer1'
    _reg_clsid_      = '{8B19F050-8354-11E1-A0FE-5C260A5D15E4}'
    def __init__(self):
        self._icon = "C:\\Users\\Administrator\\mat\\icon_overlay\\icons\\1.ico"
        self._icon_id = 101 

classes =   [IconOverlay_test,]

def is_member(path, icon_id):
    try:
        conn = None
        conn = Client("\\\\.\\pipe\\test.listener", "AF_PIPE")
        conn.send(path)
        if conn.poll(3):
            reply   =   conn.recv()
            return reply == icon_id
    except Exception:
        pass
    finally:
        conn and conn.close()
    return False

def DllRegisterServer():
    print("Registering %s ......."%IconOverlay._reg_desc_)
    import winreg
    #winreg = _winreg
    for view in [winreg.KEY_WOW64_64KEY, winreg.KEY_WOW64_32KEY]:
        for cls in classes:
            with winreg.CreateKeyEx(winreg.HKEY_LOCAL_MACHINE, r"%s\%s" %
                    (REG_PATH, cls._reg_progid_), 0,
                    winreg.KEY_ALL_ACCESS|view) as hkey:
                print("    %s"%cls)
                winreg.SetValueEx(hkey, None, 0, winreg.REG_SZ, cls._reg_clsid_)
    print("Registration complete: %s" % IconOverlay._reg_desc_)

def DllUnregisterServer():
    print("Unregistering %s ......."%IconOverlay._reg_desc_)
    import winreg
    #winreg = _winreg
    for view in [winreg.KEY_WOW64_64KEY, winreg.KEY_WOW64_32KEY]:
        for cls in classes:
            try:
                _key = winreg.DeleteKeyEx(winreg.HKEY_LOCAL_MACHINE, r"%s\%s"
                        % (REG_PATH, cls._reg_progid_),
                        winreg.KEY_ALL_ACCESS|view)
            except WindowsError as err:
                if err.errno != 2:
                    raise
    print("Unregistration complete: %s" % IconOverlay._reg_desc_)

if __name__ == '__main__':
    from win32com.server import register
    register.UseCommandLine(*classes,
                            finalize_register = DllRegisterServer,
                            finalize_unregister = DllUnregisterServer)
4

1 回答 1

0

正常工作真的很痛苦,祝你好运!

我相信 Windows 将只允许注册 10 个不同的图标,并且它只适用于按字母顺序注册的前 10 个图标。你已经有10个注册了?如果您安装了 dropbox、tor​​toise-svn 等,则很容易超过 10 个,因为每个图像都算作一个条目。如果是这种情况,请尝试在名称前加上下划线或 0 以确保它获得优先权,尽管这意味着另一个图标会丢失——我认为没有办法解决这个问题。

有时windows也不知道刷新图标的状态。你运行的是哪个版本的windows?有些人比其他人更糟。我似乎记得 XP 不是很擅长这个。虽然有一些技巧可以让它更新,你可以通过 windows api 刷新窗口,但它看起来很可怕,整个资源管理器都会闪烁。我发现一个更好的方法是更改​​文件的属性。这是我使用的技巧:

import stat,os

file_att= os.stat(path)[0]
if file_att & stat.S_IWRITE:
    os.chmod(path,stat.S_IREAD)
    os.chmod(path,stat.S_IWRITE)
else:
    os.chmod(path,stat.S_IWRITE)
    os.chmod(path,stat.S_IREAD)
于 2012-11-16T17:00:59.650 回答