451 lines
19 KiB
Objective-C
451 lines
19 KiB
Objective-C
//
|
||
// AppDelegate.m
|
||
// keyBoard
|
||
//
|
||
// Created by 张伟 on 2025/10/27.
|
||
//
|
||
|
||
#import "AppDelegate.h"
|
||
//#import "KBPermissionViewController.h"
|
||
#import <AFNetworking/AFNetworking.h>
|
||
#if !DEBUG
|
||
#import <Bugly/Bugly.h>
|
||
#endif
|
||
#import "BaseTabBarController.h"
|
||
//#import "LoginViewController.h"
|
||
//#import "KBLoginSheetViewController.h"
|
||
//#import "AppleSignInManager.h"
|
||
//#import <objc/message.h>
|
||
#import "KBULBridgeNotification.h" // Darwin 通知常量:用于通知扩展“UL已处理”
|
||
#import "LSTPopView.h"
|
||
#import "KBLoginPopView.h"
|
||
#import "KBSexSelVC.h"
|
||
#import "KBKeyboardPermissionManager.h"
|
||
#import "KBVipPay.h"
|
||
#import "KBUserSessionManager.h"
|
||
#import "KBLoginVC.h"
|
||
#import "KBConfig.h"
|
||
|
||
static NSTimeInterval const kKBSubscriptionPrefillTTL = 10 * 60.0;
|
||
|
||
// 注意:用于判断系统已启用本输入法扩展的 bundle id 需与扩展 target 的
|
||
// PRODUCT_BUNDLE_IDENTIFIER 完全一致。
|
||
// 当前工程的 CustomKeyboard target 为 com.loveKey.nyx.CustomKeyboard
|
||
//static NSString * const kKBKeyboardExtensionBundleId = @"com.loveKey.nyx.CustomKeyboard";
|
||
|
||
@implementation AppDelegate
|
||
|
||
|
||
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
|
||
/// 1:配置国际化(统一使用集中管理的语言列表)
|
||
[KBLocalizationManager shared].supportedLanguageCodes = KBDefaultSupportedLanguageCodes();
|
||
/// 2 : 处理token问题(包括卸载重装场景下的 token 清理)
|
||
[[KBUserSessionManager shared] bootstrapIfNeeded];
|
||
|
||
// 首次安装/升级:重置“完全访问”记录,避免继承旧安装遗留在 Keychain 中的状态
|
||
static NSString *const kKBFullAccessRecordInitializedKey = @"KBFullAccessRecordInitialized";
|
||
NSUserDefaults *ud = [NSUserDefaults standardUserDefaults];
|
||
if (![ud boolForKey:kKBFullAccessRecordInitializedKey]) {
|
||
[[KBKeyboardPermissionManager shared] resetFullAccessRecord];
|
||
[ud setBool:YES forKey:kKBFullAccessRecordInitializedKey];
|
||
[ud synchronize];
|
||
}
|
||
|
||
// 首次安装先进入性别选择页;点击 Skip 或确认后再进入主 TabBar
|
||
BOOL hasShownSexVC = [[NSUserDefaults standardUserDefaults] boolForKey:KBSexSelectShownKey];
|
||
if (hasShownSexVC) {
|
||
[self setupRootVC];
|
||
} else {
|
||
[self setupSexSelectRootVC];
|
||
}
|
||
|
||
// 主工程默认开启网络总开关(键盘扩展仍需用户允许完全访问后再行开启)
|
||
[KBNetworkManager shared].enabled = YES;
|
||
/// 获取网络权限
|
||
[self getNetJudge];
|
||
/// 触发一次简单网络请求,用于拉起系统的蜂窝数据权限弹窗
|
||
// dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
|
||
[self kb_fireStartupNetworkRequest];
|
||
// });
|
||
//
|
||
|
||
#if !DEBUG
|
||
/// Bugly
|
||
BuglyConfig *buglyConfig = [BuglyConfig new];
|
||
/// 设置GroupID进行配置
|
||
// buglyConfig.applicationGroupIdentifier = @"";
|
||
[Bugly startWithAppId:BuglyId config:buglyConfig];
|
||
#endif
|
||
|
||
return YES;
|
||
}
|
||
|
||
- (void)applicationDidBecomeActive:(UIApplication *)application{
|
||
// [self kb_checkNetworkAndShowAlertIfNeeded];
|
||
|
||
}
|
||
|
||
|
||
- (void)setupRootVC{
|
||
self.window = [[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]];
|
||
self.window.backgroundColor = [UIColor whiteColor];
|
||
[self.window makeKeyAndVisible];
|
||
|
||
// 根据当前是否已登录决定入口:有 token 进主 TabBar,否则先进入登录页
|
||
// BOOL loggedIn = [[KBUserSessionManager shared] isLoggedIn];
|
||
UIViewController *rootVC = nil;
|
||
|
||
rootVC = [[BaseTabBarController alloc] init];
|
||
|
||
self.window.rootViewController = rootVC;
|
||
}
|
||
|
||
- (void)toMainTabbarVC{
|
||
self.window = [[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]];
|
||
self.window.backgroundColor = [UIColor whiteColor];
|
||
[self.window makeKeyAndVisible];
|
||
UIViewController *rootVC = nil;
|
||
rootVC = [[BaseTabBarController alloc] init];
|
||
self.window.rootViewController = rootVC;
|
||
}
|
||
|
||
/// 首次安装时,展示性别选择页作为根控制器
|
||
- (void)setupSexSelectRootVC {
|
||
KBSexSelVC *vc = [[KBSexSelVC alloc] init];
|
||
self.window = [[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]];
|
||
self.window.backgroundColor = [UIColor whiteColor];
|
||
[self.window makeKeyAndVisible];
|
||
self.window.rootViewController = vc;
|
||
KBSexSelVC *sexVC = [KBSexSelVC new];
|
||
__weak typeof(self) weakSelf = self;
|
||
sexVC.didFinishSelectBlock = ^{
|
||
// 记录已经展示过性别选择页,下次启动直接进 TabBar
|
||
[[NSUserDefaults standardUserDefaults] setBool:YES forKey:KBSexSelectShownKey];
|
||
[[NSUserDefaults standardUserDefaults] synchronize];
|
||
// 切换根控制器到主 TabBar
|
||
[weakSelf setupRootVC];
|
||
};
|
||
self.window.rootViewController = sexVC;
|
||
}
|
||
|
||
#pragma mark - Permission presentation
|
||
|
||
|
||
#pragma mark - Deep Link
|
||
|
||
- (BOOL)application:(UIApplication *)application continueUserActivity:(NSUserActivity *)userActivity restorationHandler:(void (^)(NSArray * _Nullable))restorationHandler {
|
||
if ([userActivity.activityType isEqualToString:NSUserActivityTypeBrowsingWeb]) {
|
||
NSURL *url = userActivity.webpageURL;
|
||
if (!url) return NO;
|
||
NSString *host = url.host.lowercaseString ?: @"";
|
||
if ([host hasSuffix:@"app.tknb.net"]) {
|
||
NSString *path = url.path.lowercaseString ?: @"";
|
||
if ([path hasPrefix:@"/ul/settings"]) {
|
||
[self kb_openAppSettings];
|
||
return YES;
|
||
}
|
||
if ([path hasPrefix:@"/ul/login"]) {
|
||
// 区分普通登录与“充值”场景:通过 query 中的 entry=recharge 来判断
|
||
NSURLComponents *components = [NSURLComponents componentsWithURL:url resolvingAgainstBaseURL:NO];
|
||
NSString *entry = nil;
|
||
for (NSURLQueryItem *item in components.queryItems ?: @[]) {
|
||
if ([item.name isEqualToString:@"entry"]) {
|
||
entry = item.value;
|
||
break;
|
||
}
|
||
}
|
||
// 通知扩展:UL 已被主 App 接收,扩展可取消回退到 Scheme
|
||
CFNotificationCenterPostNotification(CFNotificationCenterGetDarwinNotifyCenter(),
|
||
(__bridge CFStringRef)KBDarwinULHandled,
|
||
NULL, NULL, true);
|
||
if ([entry isEqualToString:@"recharge"]) {
|
||
// 充值入口:拉起主 App 后进入充值相关页面(目前先做占位提示)
|
||
// [KBHUD showInfo:@"去充值"];
|
||
|
||
} else {
|
||
// 默认逻辑:登录
|
||
[self kb_presentLoginSheetIfNeeded];
|
||
}
|
||
return YES;
|
||
}
|
||
}
|
||
}
|
||
return NO;
|
||
}
|
||
|
||
// iOS 9+
|
||
- (BOOL)application:(UIApplication *)app openURL:(NSURL *)url options:(NSDictionary<UIApplicationOpenURLOptionsKey,id> *)options {
|
||
if (!url) return NO;
|
||
// 注意:已对 scheme 做了小写化,比较值也应为小写
|
||
if ([[url.scheme lowercaseString] isEqualToString:@"kbkeyboardappextension"]) {
|
||
NSString *urlHost = url.host ?: @"";
|
||
NSString *host = [urlHost lowercaseString];
|
||
if ([host isEqualToString:@"login"]) { // kbkeyboard://login
|
||
// 兜底路径:Scheme 也发一次“已处理”通知,避免扩展重复尝试
|
||
CFNotificationCenterPostNotification(CFNotificationCenterGetDarwinNotifyCenter(),
|
||
(__bridge CFStringRef)KBDarwinULHandled,
|
||
NULL, NULL, true);
|
||
[self kb_presentLoginSheetIfNeeded];
|
||
return YES;
|
||
} else if ([host isEqualToString:@"settings"]) { // kbkeyboard://settings
|
||
[self kb_openAppSettings];
|
||
return YES;
|
||
}else if ([host isEqualToString:@"recharge"]) { // kbkeyboard://recharge
|
||
NSDictionary<NSString *, NSString *> *params = [self kb_queryParametersFromURL:url];
|
||
NSString *productId = params[@"productId"];
|
||
BOOL autoPay = NO;
|
||
NSString *autoFlag = params[@"autoPay"];
|
||
if ([autoFlag respondsToSelector:@selector(boolValue)]) {
|
||
autoPay = autoFlag.boolValue;
|
||
}
|
||
NSString *action = params[@"action"].lowercaseString;
|
||
if ([action isEqualToString:@"autopay"]) {
|
||
autoPay = YES;
|
||
}
|
||
|
||
BOOL wantsPrefill = NO;
|
||
NSString *prefillFlag = params[@"prefill"];
|
||
if ([prefillFlag respondsToSelector:@selector(boolValue)] && prefillFlag.boolValue) {
|
||
wantsPrefill = YES;
|
||
}
|
||
NSString *src = params[@"src"];
|
||
if ([src isKindOfClass:NSString.class] && [src.lowercaseString isEqualToString:@"keyboard"]) {
|
||
wantsPrefill = YES;
|
||
}
|
||
NSDictionary *prefillPayload = wantsPrefill ? [self kb_consumeSubscriptionPrefillPayloadIfValid] : nil;
|
||
if ([prefillPayload isKindOfClass:NSDictionary.class]) {
|
||
NSString *payloadProductId = prefillPayload[@"productId"];
|
||
if (productId.length == 0 && [payloadProductId isKindOfClass:NSString.class]) {
|
||
productId = payloadProductId;
|
||
}
|
||
}
|
||
|
||
KBVipPay *vc = [[KBVipPay alloc] init];
|
||
if ([prefillPayload isKindOfClass:NSDictionary.class]) {
|
||
NSArray *productsJSON = prefillPayload[@"products"];
|
||
NSNumber *selectedIndexNumber = prefillPayload[@"selectedIndex"];
|
||
NSInteger selectedIndex = [selectedIndexNumber respondsToSelector:@selector(integerValue)] ? selectedIndexNumber.integerValue : NSNotFound;
|
||
[vc configureWithProductId:productId
|
||
autoPurchase:autoPay
|
||
prefillProductsJSON:productsJSON
|
||
selectedIndex:selectedIndex];
|
||
} else {
|
||
[vc configureWithProductId:productId autoPurchase:autoPay];
|
||
}
|
||
[KB_CURRENT_NAV pushViewController:vc animated:true];
|
||
return YES;
|
||
}
|
||
return NO;
|
||
}
|
||
return NO;
|
||
}
|
||
|
||
- (nullable NSDictionary *)kb_consumeSubscriptionPrefillPayloadIfValid {
|
||
NSUserDefaults *ud = [[NSUserDefaults alloc] initWithSuiteName:AppGroup];
|
||
if (!ud) { return nil; }
|
||
id obj = [ud objectForKey:AppGroup_SubscriptionPrefillPayload];
|
||
[ud removeObjectForKey:AppGroup_SubscriptionPrefillPayload];
|
||
[ud synchronize];
|
||
if (![obj isKindOfClass:NSDictionary.class]) { return nil; }
|
||
NSDictionary *payload = (NSDictionary *)obj;
|
||
NSNumber *ts = payload[@"ts"];
|
||
if (![ts respondsToSelector:@selector(doubleValue)]) { return nil; }
|
||
NSTimeInterval age = [NSDate date].timeIntervalSince1970 - ts.doubleValue;
|
||
if (age < 0 || age > kKBSubscriptionPrefillTTL) { return nil; }
|
||
id products = payload[@"products"];
|
||
if (![products isKindOfClass:NSArray.class] || ((NSArray *)products).count == 0) { return nil; }
|
||
return payload;
|
||
}
|
||
|
||
- (NSDictionary<NSString *, NSString *> *)kb_queryParametersFromURL:(NSURL *)url {
|
||
if (!url) { return @{}; }
|
||
NSURLComponents *components = [NSURLComponents componentsWithURL:url resolvingAgainstBaseURL:NO];
|
||
NSArray<NSURLQueryItem *> *items = components.queryItems ?: @[];
|
||
if (items.count == 0) { return @{}; }
|
||
NSMutableDictionary<NSString *, NSString *> *dict = [NSMutableDictionary dictionaryWithCapacity:items.count];
|
||
for (NSURLQueryItem *item in items) {
|
||
if (item.name.length == 0) { continue; }
|
||
dict[item.name] = item.value ?: @"";
|
||
}
|
||
return dict;
|
||
}
|
||
|
||
- (void)kb_presentLoginSheetIfNeeded {
|
||
// 已登录则不提示
|
||
// BOOL loggedIn = ([AppleSignInManager shared].storedUserIdentifier.length > 0);
|
||
// if (loggedIn) return;
|
||
// UIViewController *top = [UIViewController kb_topMostViewController];
|
||
// if (!top) return;
|
||
// [KBLoginSheetViewController presentIfNeededFrom:top];
|
||
[self goLogin];
|
||
|
||
}
|
||
|
||
- (void)goLogin{
|
||
KBLoginVC *vc = [[KBLoginVC alloc] init];
|
||
[KB_CURRENT_NAV pushViewController:vc animated:true];
|
||
// dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.25 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
|
||
// KBLoginPopView *view = [[KBLoginPopView alloc] initWithFrame:CGRectMake(0, 0, KB_SCREEN_WIDTH, KB_SCREEN_WIDTH)];
|
||
// // 创建并弹出
|
||
// LSTPopView *pop = [LSTPopView initWithCustomView:view
|
||
// parentView:nil
|
||
// popStyle:LSTPopStyleSmoothFromBottom
|
||
// dismissStyle:LSTDismissStyleSmoothToBottom];
|
||
// pop.hemStyle = LSTHemStyleBottom;
|
||
// pop.bgColor = [[UIColor blackColor] colorWithAlphaComponent:0.4];
|
||
// pop.isClickBgDismiss = YES; // 点击背景关闭
|
||
// pop.cornerRadius = 0; // 自定义 view 自处理圆角
|
||
//
|
||
// __weak typeof(pop) weakPop = pop;
|
||
// view.appleLoginHandler = ^{
|
||
// [weakPop dismiss];
|
||
// // 交给 VM 统一处理 Apple 登录 + 服务端登录
|
||
// [[KBLoginVM shared] signInWithAppleFromViewController:KB_CURRENT_NAV completion:^(BOOL success, NSError * _Nullable error) {
|
||
// if (success) {
|
||
// [KBHUD showInfo:KBLocalized(@"Signed in successfully")];
|
||
// } else {
|
||
// NSString *msg = error.localizedDescription ?: KBLocalized(@"Sign-in failed");
|
||
// [KBHUD showInfo:msg];
|
||
// }
|
||
// }];
|
||
// };
|
||
// view.closeHandler = ^{ [weakPop dismiss]; };
|
||
//
|
||
// [pop pop];
|
||
// });
|
||
|
||
|
||
}
|
||
|
||
- (void)kb_openAppSettings {
|
||
NSURL *url = [NSURL URLWithString:UIApplicationOpenSettingsURLString];
|
||
UIApplication *app = [UIApplication sharedApplication];
|
||
if ([app canOpenURL:url]) {
|
||
if (@available(iOS 10.0, *)) {
|
||
[app openURL:url options:@{} completionHandler:nil];
|
||
} else {
|
||
[app openURL:url];
|
||
}
|
||
}
|
||
}
|
||
|
||
- (void)kb_presentPermissionIfNeeded
|
||
{
|
||
// 该逻辑已迁移到 KBGuideVC,保留占位以兼容旧调用,但不再执行任何操作
|
||
}
|
||
|
||
/// 使用系统网络栈发起一次简单请求(例如拉起蜂窝数据权限弹窗)。
|
||
/// 说明:只要真正访问网络,系统在需要时就会弹出“是否允许使用蜂窝数据”等提示。
|
||
- (void)kb_fireStartupNetworkRequest {
|
||
// [[KBNetworkManager shared] GET:@"https://www.apple.com" parameters:nil headers:nil completion:^(id _Nullable jsonOrData, NSURLResponse * _Nullable response, NSError * _Nullable error) {
|
||
//
|
||
// }];
|
||
NSURL *url = [NSURL URLWithString:@"https://www.apple.com"];
|
||
if (!url) { return; }
|
||
|
||
NSURLSessionConfiguration *config = [NSURLSessionConfiguration defaultSessionConfiguration];
|
||
NSURLSession *session = [NSURLSession sessionWithConfiguration:config];
|
||
NSURLSessionDataTask *task = [session dataTaskWithURL:url
|
||
completionHandler:^(NSData * _Nullable data,
|
||
NSURLResponse * _Nullable response,
|
||
NSError * _Nullable error) {
|
||
// 回到主线程弹出一个简单提示,证明网络请求已经发起
|
||
dispatch_async(dispatch_get_main_queue(), ^{
|
||
});
|
||
}];
|
||
[task resume];
|
||
}
|
||
|
||
|
||
-(void)getNetJudge {
|
||
AFNetworkReachabilityManager *netManager = [AFNetworkReachabilityManager sharedManager];
|
||
[netManager startMonitoring];
|
||
__weak typeof(self) weakSelf = self;
|
||
[netManager setReachabilityStatusChangeBlock:^(AFNetworkReachabilityStatus status){
|
||
if (status == AFNetworkReachabilityStatusReachableViaWiFi){
|
||
// [PublicObj saveNetReachability:@"wifi"];
|
||
}else if (status == AFNetworkReachabilityStatusReachableViaWWAN){
|
||
// [PublicObj saveNetReachability:@"wwan"];
|
||
}else{
|
||
// [PublicObj saveNetReachability:@"unknown"];
|
||
}
|
||
|
||
// 网络状态变化时检查一次,并在无网络时弹出“去设置 / 取消”提示框
|
||
__strong typeof(weakSelf) strongSelf = weakSelf;
|
||
if (!strongSelf) { return; }
|
||
[strongSelf kb_checkNetworkAndShowAlertIfNeeded];
|
||
}];
|
||
}
|
||
|
||
|
||
//static BOOL KBIsKeyboardEnabled(void) {
|
||
// for (UITextInputMode *mode in [UITextInputMode activeInputModes]) {
|
||
// NSString *identifier = nil;
|
||
// @try {
|
||
// identifier = [mode valueForKey:@"identifier"]; // not a public API
|
||
// } @catch (__unused NSException *e) {
|
||
// identifier = nil;
|
||
// }
|
||
// if ([identifier isKindOfClass:[NSString class]] &&
|
||
// [identifier rangeOfString:kKBKeyboardExtensionBundleId].location != NSNotFound) {
|
||
// return YES;
|
||
// }
|
||
// }
|
||
// return NO;
|
||
//}
|
||
|
||
#pragma mark - Network check & alert
|
||
|
||
- (void)kb_checkNetworkAndShowAlertIfNeeded {
|
||
// 首次安装且尚未完成首页引导时,不弹“无网络”提示,避免在系统蜂窝数据权限弹窗前就打扰用户
|
||
static NSString *const kKBHasLaunchedOnce = @"KBHasLaunchedOnce";
|
||
NSUserDefaults *ud = [NSUserDefaults standardUserDefaults];
|
||
if (![ud boolForKey:kKBHasLaunchedOnce]) {
|
||
return;
|
||
}
|
||
|
||
// 仅在应用处于前台时提示
|
||
if ([UIApplication sharedApplication].applicationState != UIApplicationStateActive) {
|
||
return;
|
||
}
|
||
|
||
AFNetworkReachabilityStatus status = [AFNetworkReachabilityManager sharedManager].networkReachabilityStatus;
|
||
|
||
// 只有在“明确不可达”时才弹
|
||
if (status == AFNetworkReachabilityStatusNotReachable ||
|
||
status == AFNetworkReachabilityStatusUnknown) {
|
||
[self kb_showNoNetworkAlert];
|
||
}
|
||
}
|
||
|
||
- (void)kb_showNoNetworkAlert {
|
||
UIViewController *rootVC = self.window.rootViewController;
|
||
if (!rootVC) return;
|
||
|
||
// 防止重复弹(例如已经有一个 alert 在上面了)
|
||
if ([rootVC.presentedViewController isKindOfClass:[UIAlertController class]]) {
|
||
return;
|
||
}
|
||
|
||
UIAlertController *alert = [UIAlertController alertControllerWithTitle:KBLocalized(@"Network unavailable")
|
||
message:KBLocalized(@"Please check this app's wireless-data permission or network connection in Settings.")
|
||
preferredStyle:UIAlertControllerStyleAlert];
|
||
|
||
UIAlertAction *settings = [UIAlertAction actionWithTitle:KBLocalized(@"Open Settings")
|
||
style:UIAlertActionStyleDefault
|
||
handler:^(UIAlertAction * _Nonnull action) {
|
||
[self kb_openAppSettings]; // 你已经实现好的跳设置方法
|
||
}];
|
||
|
||
UIAlertAction *cancel = [UIAlertAction actionWithTitle:KBLocalized(@"Cancel")
|
||
style:UIAlertActionStyleCancel
|
||
handler:nil];
|
||
|
||
[alert addAction:settings];
|
||
[alert addAction:cancel];
|
||
|
||
[rootVC presentViewController:alert animated:YES completion:nil];
|
||
}
|
||
|
||
@end
|