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