first commit
This commit is contained in:
302
myenv/lib/python3.10/site-packages/pynput/_util/darwin.py
Normal file
302
myenv/lib/python3.10/site-packages/pynput/_util/darwin.py
Normal file
@@ -0,0 +1,302 @@
|
||||
# coding=utf-8
|
||||
# pynput
|
||||
# Copyright (C) 2015-2024 Moses Palmér
|
||||
#
|
||||
# This program is free software: you can redistribute it and/or modify it under
|
||||
# the terms of the GNU Lesser General Public License as published by the Free
|
||||
# Software Foundation, either version 3 of the License, or (at your option) any
|
||||
# later version.
|
||||
#
|
||||
# This program is distributed in the hope that it will be useful, but WITHOUT
|
||||
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
||||
# FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
|
||||
# details.
|
||||
#
|
||||
# You should have received a copy of the GNU Lesser General Public License
|
||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
"""
|
||||
Utility functions and classes for the *Darwin* backend.
|
||||
"""
|
||||
|
||||
# pylint: disable=C0103
|
||||
# pylint: disable=R0903
|
||||
# This module contains wrapper classes
|
||||
|
||||
import contextlib
|
||||
import ctypes
|
||||
import ctypes.util
|
||||
import six
|
||||
|
||||
import objc
|
||||
import HIServices
|
||||
|
||||
from CoreFoundation import (
|
||||
CFRelease
|
||||
)
|
||||
|
||||
from Quartz import (
|
||||
CFMachPortCreateRunLoopSource,
|
||||
CFRunLoopAddSource,
|
||||
CFRunLoopGetCurrent,
|
||||
CFRunLoopRunInMode,
|
||||
CFRunLoopStop,
|
||||
CGEventGetIntegerValueField,
|
||||
CGEventTapCreate,
|
||||
CGEventTapEnable,
|
||||
kCFRunLoopDefaultMode,
|
||||
kCFRunLoopRunTimedOut,
|
||||
kCGEventSourceUnixProcessID,
|
||||
kCGEventTapOptionDefault,
|
||||
kCGEventTapOptionListenOnly,
|
||||
kCGHeadInsertEventTap,
|
||||
kCGSessionEventTap)
|
||||
|
||||
|
||||
from . import AbstractListener
|
||||
|
||||
|
||||
def _wrap_value(value):
|
||||
"""Converts a pointer to a *Python objc* value.
|
||||
|
||||
:param value: The pointer to convert.
|
||||
|
||||
:return: a wrapped value
|
||||
"""
|
||||
return objc.objc_object(c_void_p=value) if value is not None else None
|
||||
|
||||
|
||||
@contextlib.contextmanager
|
||||
def _wrapped(value):
|
||||
"""A context manager that converts a raw pointer to a *Python objc* value.
|
||||
|
||||
When the block is exited, the value is released.
|
||||
|
||||
:param value: The raw value to wrap.
|
||||
"""
|
||||
wrapped_value = _wrap_value(value)
|
||||
|
||||
try:
|
||||
yield value
|
||||
finally:
|
||||
CFRelease(wrapped_value)
|
||||
|
||||
|
||||
class CarbonExtra(object):
|
||||
"""A class exposing some missing functionality from *Carbon* as class
|
||||
attributes.
|
||||
"""
|
||||
_Carbon = ctypes.cdll.LoadLibrary(ctypes.util.find_library('Carbon'))
|
||||
|
||||
_Carbon.TISCopyCurrentKeyboardInputSource.argtypes = []
|
||||
_Carbon.TISCopyCurrentKeyboardInputSource.restype = ctypes.c_void_p
|
||||
|
||||
_Carbon.TISCopyCurrentASCIICapableKeyboardLayoutInputSource.argtypes = []
|
||||
_Carbon.TISCopyCurrentASCIICapableKeyboardLayoutInputSource.restype = \
|
||||
ctypes.c_void_p
|
||||
|
||||
_Carbon.TISGetInputSourceProperty.argtypes = [
|
||||
ctypes.c_void_p, ctypes.c_void_p]
|
||||
_Carbon.TISGetInputSourceProperty.restype = ctypes.c_void_p
|
||||
|
||||
_Carbon.LMGetKbdType.argtypes = []
|
||||
_Carbon.LMGetKbdType.restype = ctypes.c_uint32
|
||||
|
||||
_Carbon.UCKeyTranslate.argtypes = [
|
||||
ctypes.c_void_p,
|
||||
ctypes.c_uint16,
|
||||
ctypes.c_uint16,
|
||||
ctypes.c_uint32,
|
||||
ctypes.c_uint32,
|
||||
ctypes.c_uint32,
|
||||
ctypes.POINTER(ctypes.c_uint32),
|
||||
ctypes.c_uint8,
|
||||
ctypes.POINTER(ctypes.c_uint8),
|
||||
ctypes.c_uint16 * 4]
|
||||
_Carbon.UCKeyTranslate.restype = ctypes.c_uint32
|
||||
|
||||
TISCopyCurrentKeyboardInputSource = \
|
||||
_Carbon.TISCopyCurrentKeyboardInputSource
|
||||
|
||||
TISCopyCurrentASCIICapableKeyboardLayoutInputSource = \
|
||||
_Carbon.TISCopyCurrentASCIICapableKeyboardLayoutInputSource
|
||||
|
||||
kTISPropertyUnicodeKeyLayoutData = ctypes.c_void_p.in_dll(
|
||||
_Carbon, 'kTISPropertyUnicodeKeyLayoutData')
|
||||
|
||||
TISGetInputSourceProperty = \
|
||||
_Carbon.TISGetInputSourceProperty
|
||||
|
||||
LMGetKbdType = \
|
||||
_Carbon.LMGetKbdType
|
||||
|
||||
kUCKeyActionDisplay = 3
|
||||
kUCKeyTranslateNoDeadKeysBit = 0
|
||||
|
||||
UCKeyTranslate = \
|
||||
_Carbon.UCKeyTranslate
|
||||
|
||||
|
||||
@contextlib.contextmanager
|
||||
def keycode_context():
|
||||
"""Returns an opaque value representing a context for translating keycodes
|
||||
to strings.
|
||||
"""
|
||||
keyboard_type, layout_data = None, None
|
||||
for source in [
|
||||
CarbonExtra.TISCopyCurrentKeyboardInputSource,
|
||||
CarbonExtra.TISCopyCurrentASCIICapableKeyboardLayoutInputSource]:
|
||||
with _wrapped(source()) as keyboard:
|
||||
keyboard_type = CarbonExtra.LMGetKbdType()
|
||||
layout = _wrap_value(CarbonExtra.TISGetInputSourceProperty(
|
||||
keyboard,
|
||||
CarbonExtra.kTISPropertyUnicodeKeyLayoutData))
|
||||
layout_data = layout.bytes().tobytes() if layout else None
|
||||
if keyboard is not None and layout_data is not None:
|
||||
break
|
||||
yield (keyboard_type, layout_data)
|
||||
|
||||
|
||||
def keycode_to_string(context, keycode, modifier_state=0):
|
||||
"""Converts a keycode to a string.
|
||||
"""
|
||||
LENGTH = 4
|
||||
|
||||
keyboard_type, layout_data = context
|
||||
|
||||
dead_key_state = ctypes.c_uint32()
|
||||
length = ctypes.c_uint8()
|
||||
unicode_string = (ctypes.c_uint16 * LENGTH)()
|
||||
CarbonExtra.UCKeyTranslate(
|
||||
layout_data,
|
||||
keycode,
|
||||
CarbonExtra.kUCKeyActionDisplay,
|
||||
modifier_state,
|
||||
keyboard_type,
|
||||
CarbonExtra.kUCKeyTranslateNoDeadKeysBit,
|
||||
ctypes.byref(dead_key_state),
|
||||
LENGTH,
|
||||
ctypes.byref(length),
|
||||
unicode_string)
|
||||
return u''.join(
|
||||
six.unichr(unicode_string[i])
|
||||
for i in range(length.value))
|
||||
|
||||
|
||||
def get_unicode_to_keycode_map():
|
||||
"""Returns a mapping from unicode strings to virtual key codes.
|
||||
|
||||
:return: a dict mapping key codes to strings
|
||||
"""
|
||||
with keycode_context() as context:
|
||||
return {
|
||||
keycode_to_string(context, keycode): keycode
|
||||
for keycode in range(128)}
|
||||
|
||||
|
||||
class ListenerMixin(object):
|
||||
"""A mixin for *Quartz* event listeners.
|
||||
|
||||
Subclasses should set a value for :attr:`_EVENTS` and implement
|
||||
:meth:`_handle_message`.
|
||||
"""
|
||||
#: The events that we listen to
|
||||
_EVENTS = tuple()
|
||||
|
||||
#: Whether this process is trusted to monitor input events.
|
||||
IS_TRUSTED = False
|
||||
|
||||
def _run(self):
|
||||
self.IS_TRUSTED = HIServices.AXIsProcessTrusted()
|
||||
if not self.IS_TRUSTED:
|
||||
self._log.warning(
|
||||
'This process is not trusted! Input event monitoring will not '
|
||||
'be possible until it is added to accessibility clients.')
|
||||
|
||||
self._loop = None
|
||||
try:
|
||||
tap = self._create_event_tap()
|
||||
if tap is None:
|
||||
self._mark_ready()
|
||||
return
|
||||
|
||||
loop_source = CFMachPortCreateRunLoopSource(
|
||||
None, tap, 0)
|
||||
self._loop = CFRunLoopGetCurrent()
|
||||
|
||||
CFRunLoopAddSource(
|
||||
self._loop, loop_source, kCFRunLoopDefaultMode)
|
||||
CGEventTapEnable(tap, True)
|
||||
|
||||
self._mark_ready()
|
||||
|
||||
# pylint: disable=W0702; we want to silence errors
|
||||
try:
|
||||
while self.running:
|
||||
result = CFRunLoopRunInMode(
|
||||
kCFRunLoopDefaultMode, 1, False)
|
||||
try:
|
||||
if result != kCFRunLoopRunTimedOut:
|
||||
break
|
||||
except AttributeError:
|
||||
# This happens during teardown of the virtual machine
|
||||
break
|
||||
|
||||
except:
|
||||
# This exception will have been passed to the main thread
|
||||
pass
|
||||
# pylint: enable=W0702
|
||||
|
||||
finally:
|
||||
self._loop = None
|
||||
|
||||
def _stop_platform(self):
|
||||
# The base class sets the running flag to False; this will cause the
|
||||
# loop around run loop invocations to terminate and set this event
|
||||
try:
|
||||
if self._loop is not None:
|
||||
CFRunLoopStop(self._loop)
|
||||
except AttributeError:
|
||||
# The loop may not have been created
|
||||
pass
|
||||
|
||||
def _create_event_tap(self):
|
||||
"""Creates the event tap used by the listener.
|
||||
|
||||
:return: an event tap
|
||||
"""
|
||||
return CGEventTapCreate(
|
||||
kCGSessionEventTap,
|
||||
kCGHeadInsertEventTap,
|
||||
kCGEventTapOptionListenOnly if (
|
||||
True
|
||||
and not self.suppress
|
||||
and self._intercept is None)
|
||||
else kCGEventTapOptionDefault,
|
||||
self._EVENTS,
|
||||
self._handler,
|
||||
None)
|
||||
|
||||
@AbstractListener._emitter
|
||||
def _handler(self, proxy, event_type, event, refcon):
|
||||
"""The callback registered with *macOS* for mouse events.
|
||||
|
||||
This method will call the callbacks registered on initialisation.
|
||||
"""
|
||||
# An injected event will have a Unix process ID attached
|
||||
is_injected = (CGEventGetIntegerValueField(
|
||||
event,
|
||||
kCGEventSourceUnixProcessID)) != 0
|
||||
|
||||
self._handle_message(proxy, event_type, event, refcon, is_injected)
|
||||
if self._intercept is not None:
|
||||
return self._intercept(event_type, event)
|
||||
elif self.suppress:
|
||||
return None
|
||||
|
||||
def _handle_message(self, proxy, event_type, event, refcon):
|
||||
"""The device specific callback handler.
|
||||
|
||||
This method calls the appropriate callback registered when this
|
||||
listener was created based on the event.
|
||||
"""
|
||||
raise NotImplementedError()
|
||||
Reference in New Issue
Block a user