如何在Mac上使用Python列出所有工作区中的所有窗口?

4
以下是Python 2代码,打印当前工作区中所有窗口的列表:
#!/usr/bin/python
import Quartz
for window in Quartz.CGWindowListCopyWindowInfo(Quartz.kCGWindowListOptionOnScreenOnly, Quartz.kCGNullWindowID):
    print("%s - %s" % (window['kCGWindowOwnerName'], window.get('kCGWindowName', u'Unknown').encode('ascii','ignore')))

虽然它不会打印全屏幕中的应用程序(因为它在另一个工作区),但这并不影响其功能。

如何修改上述脚本以列出所有桌面的所有窗口?

2个回答

5
以下脚本应该返回任何桌面/工作区/显示器、全屏以及详细信息(坐标、PID、标题等)的窗口信息:
#!/usr/bin/python

import Quartz
import time
from Quartz import CGWindowListCopyWindowInfo, kCGWindowListExcludeDesktopElements, kCGNullWindowID
from Foundation import NSSet, NSMutableSet

def windowList(wl):
    for v in wl:
        print ( 
        str(v.valueForKey_('kCGWindowOwnerPID') or '?').rjust(7) + 
            ' ' + str(v.valueForKey_('kCGWindowNumber') or '?').rjust(5) + 
            ' {' + ('' if v.valueForKey_('kCGWindowBounds') is None else ( 
                    str(int(v.valueForKey_('kCGWindowBounds').valueForKey_('X')))     + ',' + 
                    str(int(v.valueForKey_('kCGWindowBounds').valueForKey_('Y')))     + ',' + 
                    str(int(v.valueForKey_('kCGWindowBounds').valueForKey_('Width'))) + ',' + 
                    str(int(v.valueForKey_('kCGWindowBounds').valueForKey_('Height'))) 
                ) ).ljust(21) + '}' + 
            '\t[' + ((v.valueForKey_('kCGWindowOwnerName') or '') + ']') + 
            ('' if v.valueForKey_('kCGWindowName') is None else (' ' + 
            v.valueForKey_('kCGWindowName') or '')) 
        ).encode('utf8') # remove 'encode' for Python 3.x

wl1 = CGWindowListCopyWindowInfo(kCGWindowListExcludeDesktopElements, kCGNullWindowID)
print('Move target window (or ignore)\n')
time.sleep(5)

print('PID'.rjust(7) + ' ' + 'WinID'.rjust(5) + '  ' + 'x,y,w,h'.ljust(21) + ' ' + '\t[Title] SubTitle')
print('-'.rjust(7,'-') + ' ' + '-'.rjust(5,'-') + '  ' + '-'.ljust(21,'-') + ' ' + '\t-------------------------------------------')

wl2 = CGWindowListCopyWindowInfo(kCGWindowListExcludeDesktopElements, kCGNullWindowID)

w = NSMutableSet.setWithArray_(wl1)
w.minusSet_(NSSet.setWithArray_(wl2))

wl = Quartz.CGWindowListCopyWindowInfo( Quartz.kCGWindowListOptionAll, Quartz.kCGNullWindowID)
wl = sorted(wl, key=lambda k: k.valueForKey_('kCGWindowOwnerPID'))

windowList(wl)

print('\nDetailed window information: {0}\n'.format(w))

2
关键在于使用正确的选项作为CGWindowListCopyWindowInfo的第一个参数。因此,除了使用optionOnScreenOnly属性(列出当前屏幕上的所有窗口),还需要添加excludeDesktopElements属性。

excludeDesktopElements: 排除列表中任何桌面元素的窗口,包括背景图片和桌面图标。

例如:
list = CGWindowListCopyWindowInfo(kCGWindowListExcludeDesktopElements | kCGWindowListOptionOnScreenOnly, kCGNullWindowID)

或者对于所有的窗口,也可以使用kCGWindowListOptionAll属性。

kCGWindowListOptionAll:列出所有窗口,包括屏幕上和屏幕外的窗口。当使用此选项检索列表时,relativeToWindow参数应设置为kCGNullWindowID

有关其他属性,请查看CoreGraphics中的CGWindow.h


因此,原始代码可以更改为:

#!/usr/bin/python
# Prints list of all windows.
# See: https://dev59.com/MaLia4cB1Zd3GeqPeB_A
import Quartz
for window in Quartz.CGWindowListCopyWindowInfo(Quartz.kCGWindowListOptionOnScreenOnly | Quartz.kCGWindowListExcludeDesktopElements, Quartz.kCGNullWindowID):
    print("%s - %s" % (window['kCGWindowOwnerName'], window.get('kCGWindowName', u'Unknown').encode('ascii','ignore')))

1
这不应该是按位或吗? - ManuelSchneid3r

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接