AppDelegate.swift 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. import Flutter
  2. import UIKit
  3. @main
  4. @objc class AppDelegate: FlutterAppDelegate {
  5. // 存储冷启动时的URL请求
  6. private var hasLaunched: Bool = false
  7. override func application(
  8. _ application: UIApplication,
  9. didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?
  10. ) -> Bool {
  11. GeneratedPluginRegistrant.register(with: self)
  12. let controller = window?.rootViewController as! FlutterViewController
  13. FlutterMethodChannelManager.shared.setupMethodChannels(controller: controller)
  14. // 创建事件通道
  15. let keyboardEventChannel = FlutterEventChannel(name: "keyboard_ios_events",
  16. binaryMessenger: controller.binaryMessenger)
  17. // 设置事件处理器
  18. let keyboardStreamHandler = KeyboardStreamHandler()
  19. keyboardEventChannel.setStreamHandler(keyboardStreamHandler)
  20. // 开始监听键盘切换
  21. startMonitoringKeyboardChanges()
  22. // 延迟处理
  23. DispatchQueue.main.asyncAfter(deadline: .now() + 2.5) {
  24. self.hasLaunched = true
  25. }
  26. return super.application(application, didFinishLaunchingWithOptions: launchOptions)
  27. }
  28. override func application(_ app: UIApplication, open url: URL, options: [UIApplication.OpenURLOptionsKey : Any] = [:]) -> Bool {
  29. let path = url.path
  30. if let channel = FlutterMethodChannelManager.shared.keyboardChannel {
  31. if path == "/login" {
  32. if hasLaunched {
  33. channel.invokeMethod("navigateToLogin", arguments: nil)
  34. } else {
  35. // 延迟处理
  36. DispatchQueue.main.asyncAfter(deadline: .now() + 2.5) {
  37. channel.invokeMethod("navigateToLogin", arguments: nil)
  38. }
  39. }
  40. } else if path == "/member" {
  41. if hasLaunched {
  42. channel.invokeMethod("navigateToMember", arguments: nil)
  43. } else {
  44. // 延迟处理
  45. DispatchQueue.main.asyncAfter(deadline: .now() + 2.5) {
  46. channel.invokeMethod("navigateToMember", arguments: nil)
  47. }
  48. }
  49. } else if path == "/character/market" {
  50. if hasLaunched {
  51. channel.invokeMethod("navigateToCharacterMarket", arguments: nil)
  52. } else {
  53. // 延迟处理
  54. DispatchQueue.main.asyncAfter(deadline: .now() + 2.5) {
  55. channel.invokeMethod("navigateToCharacterMarket", arguments: nil)
  56. }
  57. }
  58. } else if path == "/intimacy" {
  59. if hasLaunched {
  60. channel.invokeMethod("navigateToIntimacy", arguments: nil)
  61. } else {
  62. // 延迟处理
  63. DispatchQueue.main.asyncAfter(deadline: .now() + 3.5) {
  64. channel.invokeMethod("navigateToIntimacy", arguments: nil)
  65. }
  66. }
  67. } else if path == "/character/custom" {
  68. if hasLaunched {
  69. channel.invokeMethod("navigateToCustomCharacter", arguments: nil)
  70. } else {
  71. // 延迟处理
  72. DispatchQueue.main.asyncAfter(deadline: .now() + 2.5) {
  73. channel.invokeMethod("navigateToCustomCharacter", arguments: nil)
  74. }
  75. }
  76. }
  77. }
  78. return true
  79. }
  80. func startMonitoringKeyboardChanges() {
  81. NotificationCenter.default.addObserver(self, selector: #selector(keyboardDidChange), name: UITextInputMode.currentInputModeDidChangeNotification, object: nil)
  82. }
  83. @objc func keyboardDidChange() {
  84. // 收到通知后延迟检测
  85. DispatchQueue.main.asyncAfter(deadline: .now() + 0.3) { [weak self] in
  86. if let isCustom = self?.isCustomKeybroad() {
  87. // 发送通知
  88. NotificationCenter.default.post(name: NSNotification.Name("KeyboardDidChange"), object: nil, userInfo: ["isCustomKeyboard": isCustom])
  89. }
  90. }
  91. }
  92. func isCustomKeybroad() -> Bool {
  93. let currentKeyboardName = (((UITextInputMode.activeInputModes as NSArray).filtered(using: NSPredicate(format: "isDisplayed = YES"))).last as? NSObject)?.value(forKey: "extendedDisplayName") as? String
  94. let infoDictionary = Bundle.main.infoDictionary!
  95. let appDisplayName = infoDictionary["CFBundleDisplayName"] as? String
  96. return currentKeyboardName == appDisplayName
  97. }
  98. }
  99. // 事件流处理器
  100. class KeyboardStreamHandler: NSObject, FlutterStreamHandler {
  101. private var eventSink: FlutterEventSink?
  102. override init() {
  103. super.init()
  104. // 注册通知
  105. NotificationCenter.default.addObserver(self, selector: #selector(keyboardDidChange(_:)), name: NSNotification.Name("KeyboardDidChange"), object: nil)
  106. }
  107. @objc func keyboardDidChange(_ notification: Notification) {
  108. if let userInfo = notification.userInfo,
  109. let isCustomKeyboard = userInfo["isCustomKeyboard"] as? Bool,
  110. let eventSink = eventSink {
  111. // 发送事件到Flutter
  112. eventSink(isCustomKeyboard)
  113. }
  114. }
  115. func onListen(withArguments arguments: Any?, eventSink events: @escaping FlutterEventSink) -> FlutterError? {
  116. eventSink = events
  117. return nil
  118. }
  119. func onCancel(withArguments arguments: Any?) -> FlutterError? {
  120. eventSink = nil
  121. return nil
  122. }
  123. deinit {
  124. NotificationCenter.default.removeObserver(self)
  125. }
  126. }