窗口中的图标覆盖不反映在资源管理器中

时间:2012-11-16 16:21:51

标签: python windows explorer py2exe shell-icons

我使用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)

1 个答案:

答案 0 :(得分:0)

这对于正常工作来说真是太痛苦了,祝你好运!

我相信Windows只允许注册10个不同的图标,并且只能按字母顺序注册前10个图标。你已经注册了10个吗?如果您安装了Dropbox,tortoise-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)