关于 Mac 右键菜单

2011-10-3 Jerry Mac

Mac OS X Lion 的 Launch Service 用于关联应用程序和文件并维护最近打开的项目列表。

在文件关联和右键菜单方面,每当系统安装一个新的应用程序,都会调用 Launch Service 的 API 注册关联的文件类型。有两种情况,一种是通过 pkg 安装包安装的,通常这些应用程序会通过脚本主动向 Launch Service 注册;而第二种常见的拖拽式 .app 应用程序,则是由 Finder 和系统通过对应用程序的的一些判断后代理注册的。应用程序通过 Launch Service 注册以后,就和特定的支持文档产生了关联。

Launch Service 隶属于 Application Service Framework (包含一堆的应用程序接口,开发者可以通过这些接口,调用系统服务),用于使一个运行中的程序,能够打开另一个程序,文档,URL的接口。它可以打开另一个程序;在另一个程序中打开文档或URL;找到对于一个文档或URL最适用的程序;为一个应用程序注册它可以关联的文档类型和URL;获得一个文件,URL等正确的显示方式,比如如何显示此类文件的图标以及信息等;维护和更新最近试用过的最近试用程序和文档的列表。

从原理上看,Launch Service 维护着一个文件到应用程序之间的多对多对应关系,这个关系是存在一个数据库中。这个数据库被称作L aunch Service Database。对于 Mac OS X 下的每一个文件都有描述信息(包括我们从 GetInfo 中看到的一些)。Launch Service 感兴趣的,就是这个文件的文件类型码,创建者签名,文件扩展名,显示名称(用在 Finder 或 Dock 中显示),文件通用类型描述(比如,是应用程序,还是文件夹,或是替身,或是文件或视频)。除了这些,还有一些额外的 (Meta Data 用于快速描述文件信息) 标志位。比如,是否是可执行程序,是否是容器(文件夹,包,卷,dmg),是否是隐藏文件等等。而应用程序方面,Launch Service 会从应用程序的 info.plist 中获取诸如应用程序名称,图标,应用程序可打开的文件或URL类型,运行环境,是否有UI,对应权限等信息。Launch Service 就会根据这些信息,建立数据库,这就是右键点击文件时,看到的可打开此文件的应用程序列表。当然,如果一个文件根本没有任何匹配,右键菜单为空。Launch Service 会跳出窗口,让用户自行选择应用程序。用户选择后,Launch Service 就会将这个对应关系保存在数据库中。

Launch Service 对于一个文件关联多种应用程序的时候,也是有优先级排序规则的,从右键菜单就可以看出,Launch Service 会有一个默认选择的应用程序。它的排序规则是:
1. 用户手动指定的应用程序拥有最高优先级。
2. 如果没有指定,那么 Launch Service 会查看此文件扩展名,然后找到数据库中所有跟此扩展名相关的应用程序。
3. 如果没有扩展名,或者第二步中找到多于一个的应用程序,LS 会查找该文件是否含有文件类型码,再按照此类型码在数据库中查找所有相关应用程序。
4. 如果通过2,3步还是找到了多于一个的应用程序,那么,首先查找哪些应用程序注册的创建者签名和文件的创建者签名匹配,然后再查找哪些应用程序是否是 Mac OS X 本地应用程序(Native Program,指的是不用 Classic 模拟环境,X11 模拟环境,Java,Python 等等),再查找应用程序是否是存在于系统启动卷上,再查找哪些应用程序在本地卷上,如果到这里还是剩下多于一个的应用程序,就只能比版本号了。如果还是比不出来,那么 LS 就会随便排序了。

通常,我们在 Mac 下会遇到重复菜单项的问题,或者某些已经卸载的程序的菜单项遗留。从原因上来看,有三种可能:
1. 除了 LS 感兴趣的项目外,还有好多其他的项目也要被存储在数据库中。每当 LS 要注册一个新应用程序时,它会先看这个应用程序中有没有一个 “inUpdate” 的描述,如果此描述值为 1 (就是Yes),那么LS会查找数据库进行匹配。匹配到后会用这个新应用程序的信息来更新这个旧的信息。如果“inUpdate”为0,那么 LS 会直接注册这个应用程序并关联文件。这可能是潜在的重复项出现原因。
2. 有些装在 VM 或者 PD 的应用程序,在 VM 或 PD 卸载后也会残留到 LS 数据库中。
3. 应用程序描述的一些更新变化,使得 LS 认为这是一个全新的应用程序而直接注册。这三个方面都有可能造成重复菜单项的出现,而且很难完全避免。

如果需要清理菜单重复项和无效的关联,可以在终端运行下面命令,在本地、系统和用户空间上,重建LS数据库:
/System/Library/Frameworks/CoreServices.framework/Frameworks/LaunchServices.framework/Support/lsregister -kill -r -domain local -domain system -domain user

lsregister 命令参数如下:
-kill:重置全局LS数据库(最先执行)
-lint:打印详细应用程序文件关联注册中的错误信息
-convert:将老数据库中的信息注册到新的LS数据库
-load:加载LS插件
-lazy n:指定一个注册等待时间
-r:递归的查找文件夹内容以做关联之用(不包括pkg类型文件和隐藏文件夹下的内容)
-R:递归的查找文件夹内容以做关联之用(包括pkg类型文件和隐藏文件夹下的内容)
-f:强制更新所有对应注册信息
-v:输出lsregister运行详细信息
-dump:在注册完成后显示数据库内容
-h:显示此帮助

标签: Mac


发表评论:

Powered by emlog sitemap