Files
uskey/Sources/EventTapManager.swift
loveuer 6285cabdd1
Some checks failed
Build and Release / Build DMG for arm64 (push) Has been cancelled
Build and Release / Build DMG for x86_64 (push) Has been cancelled
Build and Release / Create Release (push) Has been cancelled
Major improvements and bug fixes for v1.1.0
Features:
- Add auto-enable on startup based on config
- Add enabled state persistence in config.json
- Add limitations notice in GUI menu
- Simplify logging to single file (uskey.log)
- Change "Open Logs Folder" to "Open Config Folder"

Bug Fixes:
- Fix GUI status display mismatch on startup
- Fix first toggle click not working issue
- Revert password field changes that caused key blocking

UI Changes:
- Add ⚠️ emoji to "About Limitations" menu item
- Remove button action to fix menu refresh

🤖 Generated with [Qoder][https://qoder.com]
2025-12-05 13:48:15 +08:00

114 lines
3.5 KiB
Swift

@preconcurrency import Cocoa
@preconcurrency import CoreGraphics
@preconcurrency import ApplicationServices
class EventTapManager {
private var eventTap: CFMachPort?
private var runLoopSource: CFRunLoopSource?
private var isEnabled: Bool = false
var keyMapper: KeyMapper
init(keyMapper: KeyMapper) {
self.keyMapper = keyMapper
}
func start() -> Bool {
guard !isEnabled else { return true }
Logger.debug("Creating event tap...")
let eventMask = (1 << CGEventType.keyDown.rawValue) | (1 << CGEventType.keyUp.rawValue)
let selfPtr = Unmanaged.passUnretained(self).toOpaque()
let callback: CGEventTapCallBack = { proxy, type, event, refcon in
guard type == .keyDown || type == .keyUp else {
return Unmanaged.passRetained(event)
}
guard let refcon = refcon else {
Logger.error("Event callback: refcon is nil")
return Unmanaged.passRetained(event)
}
let manager = Unmanaged<EventTapManager>.fromOpaque(refcon).takeUnretainedValue()
let keyCode = event.getIntegerValueField(.keyboardEventKeycode)
if manager.keyMapper.hasMappingFor(keyCode: keyCode) {
if let mappedKey = manager.keyMapper.getMappedKey(for: keyCode) {
Logger.debug("Remapping: \(keyCode) -> \(mappedKey)")
event.setIntegerValueField(.keyboardEventKeycode, value: mappedKey)
}
}
return Unmanaged.passRetained(event)
}
eventTap = CGEvent.tapCreate(
tap: .cgSessionEventTap,
place: .headInsertEventTap,
options: .defaultTap,
eventsOfInterest: CGEventMask(eventMask),
callback: callback,
userInfo: selfPtr
)
guard let eventTap = eventTap else {
Logger.error("Failed to create event tap - check accessibility permissions")
return false
}
Logger.debug("Event tap created successfully")
runLoopSource = CFMachPortCreateRunLoopSource(kCFAllocatorDefault, eventTap, 0)
guard let runLoopSource = runLoopSource else {
Logger.error("Failed to create run loop source")
return false
}
Logger.debug("Adding event tap to run loop...")
CFRunLoopAddSource(CFRunLoopGetCurrent(), runLoopSource, .commonModes)
CGEvent.tapEnable(tap: eventTap, enable: true)
isEnabled = true
Logger.info("Event tap started")
return true
}
func stop() {
guard isEnabled else { return }
Logger.debug("Stopping event tap...")
if let eventTap = eventTap {
CGEvent.tapEnable(tap: eventTap, enable: false)
if let runLoopSource = runLoopSource {
CFRunLoopRemoveSource(CFRunLoopGetCurrent(), runLoopSource, .commonModes)
}
}
eventTap = nil
runLoopSource = nil
isEnabled = false
Logger.info("Event tap stopped")
}
func isRunning() -> Bool {
return isEnabled
}
func reload(config: Config) {
let wasEnabled = isEnabled
if wasEnabled {
stop()
}
keyMapper.loadFromConfig(config)
if wasEnabled {
_ = start()
}
}
}