如何在Windows下用Python开发右键菜单功能?

本文讲述的核心库:_winreg

平时在Windows下安装一些软件后,会发现自动添加了一些右键菜单功能,方便日常使用。比如安装百度网盘客户端之后,在右键点击一个文件之后,就会发现多了一个右键菜单:上传到百度网盘,使用起来非常方便。

当然系统自带的右键菜单也有很多,那么这样的右键菜单功能是怎么实现的呢?本文就从一个简单的例子讲起,来探究一下怎么开发简单的右键菜单功能。

_winreg模块

本文实现右键菜单功能使用Pyhton内置的_winreg模块,从其名字也可以看出,操作右键菜单,实际上操作的是Windows的注册表。

右键菜单的分类

右键菜单不止有一种,比如右键点击一个文件、一个文件夹、文件夹空白处、磁盘盘符、在一个应用程序内部点击右键等等场景。下面主要讲述前四种右键菜单的添加方式。

手动修改注册表添加右键菜单实现打印"Hello World"

其实在Windows下添加右键菜单的核心步骤就是在注册表中新增一个叫注册表键的东西。在使用Python添加注册表右键菜单之前,先来尝试一下怎么通过手动修改注册表来添加右键菜单。

注册表编辑器添加右键菜单的基本原理

可以通过如下步骤初步了解右键菜单的原理:

  • win + R快捷键打开运行命令输入框,输入regedit命令并回车,打开注册表编辑器。
  • 可以看到注册表中是类似于目录树一样的树形结构,树中的每个项目都称为key,也就是注册表的键,其实我们添加右键菜单,本质上就是在某个父键下面新增一个具体执行某种操作的子键。
  • 找到这样一个键:HKEY_CLASSES_ROOT/Directory/Background/shell,这个就是添加文件夹空白处右键菜单的父键,可以看到当前该父键下已经有三个子键,也就是有三个右键菜单了:
  • 在任何一个文件夹的空白处点击右键,也可以看到对应的右键菜单,比如git_guigit_shell
  • 下面来仔细研究一下git_shell这个右键菜单子键,选择这个子键,展开,可以看到还有一个名为command的子键,还可以在注册表编辑器右边看到git_shell这个键的详情:
  • 选择git_shellcommand子键:
  • 这时终于恍然大悟,原来点击Git Bash Here这个右键菜单,原来是执行了D:\Programs\Git\git-bash.exe这个程序,并且自动带上了--cd=%v.这样一个参数。还可以右键点击名称,修改这个键的值:
  • 如果不想使用某个右键菜单了,只需把这个右键菜单的键(包括子键)在注册表编辑器中删掉即可:


添加右键菜单打印"Hello World"

相信通过上述讲解,对注册表编辑器和右键菜单的关系已经有了初步的认识,那么下面我想添加一个文件夹空白处的右键菜单项,当点击这个菜单的时候就自动打开命令行并输出一句:Hello World
详细步骤如下:

  • 在某个文件夹下,比如在D盘根目录创建一个hello.py文件,其内容如下,用于在命令行打印出Hello World
# coding:utf-8
import os
print 'Hello World'
# 防止命令行一闪而过
os.system('pause')
  • 打开注册表编辑器,在HKEY_CLASSES_ROOT/Directory/Background/shell键下面新建一个名为HelloWorld的子键(项),并为HelloWorld子键添加一个名为command的子键:

  • 选中HelloWorld并在右侧修改它的数据为:Hello World!,这也是右键菜单显示的名称:
  • 选中command并在右侧修改其数据为:python d:/hello.py
  • 不需要做任何保存操作,此时就可以直接到任意一个文件夹的空白处,点击右键,就可以发现多了一个名为Hello World!的右键菜单:
  • 点击Hello World!菜单,弹出命令行窗口,效果如下:

注意点

在注册表编辑器中添加一个右键菜单键时,执行的命令(command)只能是可执行文件或在系统环境变量中配置过的命令,不能是DOS命令。

用Python实现最简单的"Hello World"右键菜单

经过上面的手工添加右键菜单的步骤,我们已经对如何通过修改注册表的方式添加一个右键菜单的过程比较熟悉了,那么接下来能不能用Python代码自动添加一个右键菜单,当在任何一个文件夹空白处点击这个右键菜单的时候,输出一个Hello World呢?当然可以,其实就是用代码来实现我们上面的一系列手工步骤。

步骤如下:

  • 在某个文件夹下,比如在D盘根目录创建一个hello.py文件,其内容如下,用于在命令行打印出Hello World
# coding:utf-8
import os
print 'Hello World'
# 防止命令行一闪而过
os.system('pause')
  • 创建一个add_context_menu.py文件,代码如下:
# coding:utf-8
import _winreg as reg
def add_hello_context_menu():
    '''
    添加右键菜单,打印'Hello World'
    :return:
    '''
    # 菜单名称,如果含有中文,需要采用GBK编码格式,否则会出现乱码
    menu_name = 'Hello, 你好世界!'.decode('utf-8').encode('gbk')
    # 点击菜单所执行的命令
    menu_command = 'python d:/hello.py'

    # 打开名称为'HEKY_CLASSES_ROOT\\Directory\\Background\\shell'的注册表键,第一个参数为key,第二个参数为sub_key
    # 函数原型:OpenKey(key, sub_key, res = 0, sam = KEY_READ)
    # 注:路径分隔符依然要使用双斜杠'\\'
    key = reg.OpenKey(reg.HKEY_CLASSES_ROOT, r'Directory\\Background\\shell')

    # 为key创建一个名称为menu_name的sub_key,并设置sub_key的值为menu_name,数据类型为REG_SZ即字符串类型,后面跟的'(&H)'表示执行该sub_key的快捷键
    # 函数原型:SetValue(key, sub_key, type, value)
    reg.SetValue(key, menu_name, reg.REG_SZ, menu_name + '(&H)')

    # 打开刚刚创建的名为menu_name的子键
    sub_key = reg.OpenKey(key, menu_name)
    # 为sub_key添加名为'command'的子键,并设置其值为menu_command,数据类型为REG_SZ字符串类型
    reg.SetValue(sub_key, 'command', reg.REG_SZ, menu_command)

    # 关闭sub_key和key
    reg.CloseKey(sub_key)
    reg.CloseKey(key)

if __name__ == '__main__':
    add_hello_context_menu()
  • 运行上面这段代码,运行完之后打开注册表编辑器,发现多了一个名为Hello, 你好世界!的键:

    其command子键的值为:
  • 在任意一个文件夹空白处点击右键,发现多了一个名为Hello, 你好世界!的右键菜单:
  • 点击这个菜单,或按快捷键H,弹出命令行窗口如下:
  • 由此达到目标!
  • 注:可以重复运行添加同一个右键菜单的脚本,每次运行,键名相同的情况下后一次会覆盖前一次。

实现"显示文件/文件夹路径"右键菜单

经过上面一段代码,我们对如何使用Python代码添加注册表菜单的套路就有所了解了。那么能不能来点高级的,比如右键点击一个文件或文件夹时,可以显示出这个文件或文件夹的路径?当然可以,talk is cheap,show code:

  • 首先在D盘根目录下创建一个名为show_path.py的文件,用于在命令行中打印出路径(路径是作为命令行参数的第二个参数传入的)内容如下
# coding:utf-8
# 打印出命令行参数第2个参数
import sys
import os
print 'current path is: {0}'.format(sys.argv[1])
# 等待,防止命令行一闪而过
os.system('pause')
  • 添加右键菜单的代码,其中把添加右键菜单的过程做了一个封装,封装成了add_context_menu函数:
# coding:utf-8
import _winreg as reg
def add_context_menu(menu_name,command,reg_root_key_path,reg_key_path,shortcut_key):
    '''
    封装的添加一个右键菜单的方法
    :param menu_name: 显示的菜单名称
    :param command: 菜单执行的命令
    :param reg_root_key_path: 注册表根键路径
    :param reg_key_path: 要添加到的注册表父键的路径(相对路径)
    :param shortcut_key: 菜单快捷键,如:'S'
    :return:
    '''
    # 打开名称父键
    key = reg.OpenKey(reg_root_key_path, reg_key_path)
    # 为key创建一个名称为menu_name的sub_key,并设置sub_key的值为menu_name加上快捷键,数据类型为REG_SZ字符串类型
    reg.SetValue(key, menu_name, reg.REG_SZ, menu_name + '(&{0})'.format(shortcut_key))

    # 打开刚刚创建的名为menu_name的sub_key
    sub_key = reg.OpenKey(key, menu_name)
    # 为sub_key添加名为'command'的子键,并设置其值为command + ' "%v"',数据类型为REG_SZ字符串类型
    reg.SetValue(sub_key, 'command', reg.REG_SZ, command + ' "%v"')

    # 关闭sub_key和key
    reg.CloseKey(sub_key)
    reg.CloseKey(key)

def add_show_file_path_menu():
    '''
    添加右键菜单,可以在右键点击一个文件、目录、文件夹空白处或驱动器盘符后在命令行中打印出当前的绝对路径
    :return: None
    '''
    # 菜单名称
    menu_name = 'Show file path'
    # 执行一个python脚本的命令,用于打印命令行参数的第二个参数(即选中的文件路径)
    py_command = r'python D:\\show_path.py'

    # 添加文件右键菜单
 add_context_menu(menu_name,py_command,reg.HKEY_CLASSES_ROOT,r'*\\shell','S')
    # 添加文件夹右键菜单
    add_context_menu(menu_name, py_command, reg.HKEY_CLASSES_ROOT, r'Directory\\shell', 'S')
    # 添加文件夹空白处右键菜单
    add_context_menu(menu_name, py_command, reg.HKEY_CLASSES_ROOT, r'Directory\\Background\\shell', 'S')
    # 添加磁盘驱动器右键菜单
    add_context_menu(menu_name, py_command, reg.HKEY_CLASSES_ROOT, r'Drive\\shell', 'S')

if __name__ == '__main__':
    add_show_file_path_menu()
  • 运行上述代码,然后右键点击某个文件夹,可以看到多出一个名为Show file path的右键菜单:

    点击这个菜单,效果如下:
  • 再分别右键点击文件、文件夹空白处、盘符,都可以看到同样的菜单:




  • 可以看到,代码中添加注册表键的command命令时,是以%v代表一个路径参数的,还有其他几种参数可以使用:
系统默认变量的含义:
%1  表示程序操作的文件
%2  表示系统默认的打印机
%3  表示资料扇区
%4  表示操作的Port端口
"%v"  程序操作的路径

删除右键菜单

上面添加了好几个右键菜单,那假如我不想要这几个右键菜单了该怎么删除呢?有两种方法可以删:一是去注册表编辑器里手动一个个删掉,第二个方法就是用代码删除,删除上面添加的Show file path右键菜单的代码如下:

import _winreg as reg
def delete_reg_key(root_key,key,menu_name):
    '''
    删除一个右键菜单注册表子键
    :param root_key:根键
    :param key: 父键
    :param menu_name: 菜单子键名称
    :return: None
    '''
    try:
        parent_key = reg.OpenKey(root_key,key)
    except Exception as msg:
        print msg
        return
    if parent_key:
        try:
            menu_key = reg.OpenKey(parent_key,menu_name)
        except Exception as msg:
            print msg
            return
        if menu_key:
            try:
                # 必须先删除子键的子键,才能删除子键本身
                reg.DeleteKey(menu_key,'command')
            except Exception as msg:
                print msg
                return
            else:
                reg.DeleteKey(parent_key,menu_name)

if __name__ == '__main__':
    menu_name = 'Show file path'
    delete_reg_key(reg.HKEY_CLASSES_ROOT,r'*\\shell',menu_name)
    delete_reg_key(reg.HKEY_CLASSES_ROOT, r'Directory\\shell', menu_name)
    delete_reg_key(reg.HKEY_CLASSES_ROOT, r'Directory\\Background\\shell', menu_name)
    delete_reg_key(reg.HKEY_CLASSES_ROOT, r'Drive\\shell', menu_name)

运行上面代码后,发现Show file path右键菜单已经不见了,说明已经删除成功了。

实战:实现"用Chrome浏览器打开文件"的右键菜单

下面来实现一个简单的功能:在一个文件上点击右键菜单,可以出现一个用谷歌浏览器打开的右键菜单,talk is cheap,上代码:

import _winreg as reg
def add_context_menu(menu_name,command,reg_root_key_path,reg_key_path,shortcut_key):
    '''
    封装的添加一个右键菜单的方法
    :param menu_name: 显示的菜单名称
    :param command: 菜单执行的命令
    :param reg_root_key_path: 注册表根键路径
    :param reg_key_path: 要添加到的注册表父键的路径(相对路径)
    :param shortcut_key: 菜单快捷键,如:'S'
    :return:
    '''
    # 打开名称父键
    key = reg.OpenKey(reg_root_key_path, reg_key_path)
    # 为key创建一个名称为menu_name的sub_key,并设置sub_key的值为menu_name加上快捷键,数据类型为REG_SZ字符串类型
    reg.SetValue(key, menu_name, reg.REG_SZ, menu_name + '(&{0})'.format(shortcut_key))

    # 打开刚刚创建的名为menu_name的sub_key
    sub_key = reg.OpenKey(key, menu_name)
    # 为sub_key添加名为'command'的子键,并设置其值为command + ' "%v"',数据类型为REG_SZ字符串类型
    reg.SetValue(sub_key, 'command', reg.REG_SZ, command + ' "%v"')

    # 关闭sub_key和key
    reg.CloseKey(sub_key)
    reg.CloseKey(key)

def add_open_with_chrome():
    '''
    添加"用谷歌浏览器打开"右键菜单
    :return:
    '''
    # 右键菜单名
    menu_name = 'Open with chrome'
    # Chrome浏览器可执行文件的本地绝对路径
    command = r'C:\\Users\\Administrator.PC-20170728DWIF\\AppData\\Local\\Google\\Chrome\\Application\\chrome.exe'
    # 注册表根键
    reg_root_key_path = reg.HKEY_CLASSES_ROOT
    # 注册表父键
    reg_key_path = r'*\\shell'
    # 快捷键
    shortcut_key = 'C'
    add_context_menu(menu_name, command, reg_root_key_path, reg_key_path, shortcut_key)

if __name__ == '__main__':
    add_open_with_chrome()

运行上述代码,然后右键点击一个文本文件,可以看到新增的菜单:


点击Open with chrome(C)菜单,可以看到用谷歌浏览器打开了hello.py这个文本文件:

本文源代码GitHub路径

本文涉及的源代码都已经提交到本人的GitHub

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,047评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,807评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,501评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,839评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,951评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,117评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,188评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,929评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,372评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,679评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,837评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,536评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,168评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,886评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,129评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,665评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,739评论 2 351

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,642评论 18 139
  • Ubuntu的发音 Ubuntu,源于非洲祖鲁人和科萨人的语言,发作 oo-boon-too 的音。了解发音是有意...
    萤火虫de梦阅读 99,217评论 9 467
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,887评论 25 707
  • 武德九年,唐太宗李世民发动玄武门之变,成为太子,28岁登基,在位23年。他接受隋亡的教训,励精图治,与了休养生息,...
    悠悠千古事阅读 339评论 0 0
  • 我想你们记得的,我的童童。 那时挂在嘴边,是她贴在心尖。 现在,在哪里。 渐渐发现别人不屑的点滴,比如我要买做窗帘...
    唐四月阅读 164评论 0 0