AppDelegate.m 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503
  1. //
  2. // AppDelegate.m
  3. // 唔即云相册
  4. //
  5. // Created by 余衡武 on 2021/12/8.
  6. //
  7. #import "AppDelegate.h"
  8. #import <CoreMotion/CoreMotion.h>
  9. #import "ShearDeviceUDPManager.h"
  10. #import "AudioSessionObject.h"
  11. #import "DDYLanguageTool.h"
  12. #import "PLeakSniffer.h"
  13. #import "connectDeviceManager.h"
  14. #import <Bugly/Bugly.h>
  15. #import <JJException/JJException.h>
  16. #import <WXApi.h>
  17. #import <TencentOpenAPI/QQApiInterface.h>
  18. #import <TencentOpenAPI/TencentOAuth.h>
  19. #import <SDWebImage/SDWebImage.h>
  20. //#import <ZFPlayer/ZFLandscapeRotationManager.h>
  21. #import "ZFLandscapeRotationManager.h"
  22. #import <WebRTC/AMediaStream.h>
  23. #import "pingManager.h"
  24. #import "nasUploadFileManager.h"
  25. @interface AppDelegate ()<JJExceptionHandle,WXApiDelegate>
  26. {
  27. CMMotionManager *cmManager;
  28. NSString * NASFileByBoxServiceByPingok;//内网地址
  29. NSString * NASFileByBoxServiceByPingNot;//外网地址
  30. }
  31. @end
  32. @implementation AppDelegate
  33. + (AppDelegate*)sharedAppDelegate
  34. {
  35. static AppDelegate *appDelegate = nil;
  36. static dispatch_once_t onceToken;
  37. dispatch_once(&onceToken, ^{
  38. if (appDelegate == nil) {
  39. appDelegate = (AppDelegate*)[[UIApplication sharedApplication] delegate];
  40. }
  41. });
  42. return appDelegate;
  43. }
  44. - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
  45. [[UIApplication sharedApplication] setStatusBarStyle:UIStatusBarStyleDarkContent];
  46. //帮客户生成秘钥
  47. // NSString *snStr = @"0333933700222450011053";
  48. // NSString *secretkey = [RSATool sha256_8:snStr];
  49. // HLog(@"%@",secretkey);
  50. [AMediaStream globalInitialization];
  51. //保护App,一般常见的问题不会导致闪退,增强App的健壮性,同时会将错误抛出来,根据每个App自身的日志渠道记录
  52. [JJException configExceptionCategory:JJExceptionGuardAll];
  53. [JJException startGuardException];
  54. [JJException registerExceptionHandle:self];
  55. // //Default value:NO no表示异常不退出 YES 表示退出 开发时应该设置为YES
  56. // JJException.exceptionWhenTerminate = YES;
  57. //test code
  58. // NSArray * arr = @[];
  59. // NSString *str = arr[2];
  60. [self setLanguagesFun];
  61. //设置默认值
  62. self.couldPhone_W_PHONE = 720.0;
  63. self.couldPhone_H_PHONE = 1280.0;
  64. // self.couldPhone_W_PHONE = 1080.0;
  65. // self.couldPhone_H_PHONE = 1920.0;
  66. [[UIButton appearance] setExclusiveTouch:YES];
  67. cmManager = [[CMMotionManager alloc] init];
  68. if (cmManager.isAccelerometerAvailable){
  69. HLog(@"\n-------摇一摇可用-----");
  70. }else{
  71. HLog(@"\n-------摇一摇不可用-----");
  72. }
  73. cmManager.accelerometerUpdateInterval = 0.1;
  74. [cmManager startAccelerometerUpdates];
  75. [cmManager startDeviceMotionUpdatesToQueue:[NSOperationQueue new] withHandler:^(CMDeviceMotion * _Nullable motion, NSError * _Nullable error) {
  76. // CMRotationRate rotationRate = motion.rotationRate;
  77. // CGFloat rotationRatex = rotationRate.x;
  78. // CGFloat rotationRatey = rotationRate.y;
  79. // CGFloat rotationRatez = rotationRate.z;
  80. CMAcceleration accrleration = motion.gravity;
  81. CGFloat rotationGravityz = accrleration.z;
  82. //HLog(@"rotationRatey: %f",rotationRatey)
  83. //HLog(@"rotationGravityz: %f",rotationGravityz)
  84. //if (rotationRatey > 7){
  85. if (rotationGravityz > 0.85){
  86. BOOL haveOpenMask = [HWDataManager getBoolWithKey:Consn_Fanzhuan_Exit_app_Open];
  87. BOOL isPrivacyMode = ksharedAppDelegate.DeviceThirdIdMod.data.isPrivacyMode;
  88. if (haveOpenMask && isPrivacyMode){
  89. exit(0);/*强制退出app*/
  90. }
  91. }
  92. }];
  93. // [[ShearDeviceUDPManager shareInstance] startShearchDevice];
  94. // [[ShearDeviceUDPManager shareInstance] shearchDeviceLoop];
  95. //启动后台保活
  96. [AudioSessionObject shareManager];
  97. [self MonitorNetworkChangesFun];
  98. #ifdef DEBUG
  99. [[PLeakSniffer sharedInstance] installLeakSniffer];
  100. #else
  101. [Bugly startWithAppId:@"179559a521"];
  102. #endif
  103. [SVProgressHUD setDefaultStyle:(SVProgressHUDStyleDark)];
  104. [[UIDevice currentDevice] setBatteryMonitoringEnabled:YES];
  105. [[UIDevice currentDevice] batteryLevel];
  106. // 监听电池电量变化通知
  107. //[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(batteryLevelChanged:) name:UIDeviceBatteryLevelDidChangeNotification object:nil];
  108. //微信注册
  109. [WXApi registerApp:WXAPPid universalLink:wxuniversalLink];
  110. [TencentOAuth setIsUserAgreedAuthorization:YES];
  111. TencentOAuth *tencentOAuth =[[TencentOAuth alloc] initWithAppId:QQAPPid andUniversalLink:QQUniversalLink andDelegate:self];
  112. [self imageLoadingSettings];
  113. //写WebRct日志到本地
  114. NSString *logFilePath = [kSHPath_logs stringByAppendingPathComponent:@"iOSWebRtc.log"];
  115. // 创建或打开文件
  116. NSFileManager *fileManager = [NSFileManager defaultManager];
  117. if (![fileManager fileExistsAtPath:logFilePath]) {
  118. // 如果文件不存在,则创建它
  119. if (![[NSFileManager defaultManager] createFileAtPath:logFilePath contents:nil attributes:nil]) {
  120. HLog(@"Unable to create file: %@", logFilePath);
  121. }
  122. }
  123. //_WebRtcLogger = [[RTC_OBJC_TYPE(RTCFileLogger) alloc] initWithDirPath:logFilePath maxFileSize:10*1024*1024];
  124. _WebRtcLogger = [[RTCFileLogger alloc] initWithDirPath:kSHPath_logs maxFileSize:10*1024*1024];
  125. HLog(@"_WebRtcLogger: %@", _WebRtcLogger);
  126. return YES;
  127. }
  128. /// 在这里写支持的旋转方向,为了防止横屏方向,应用启动时候界面变为横屏模式
  129. - (UIInterfaceOrientationMask)application:(UIApplication *)application supportedInterfaceOrientationsForWindow:(UIWindow *)window {
  130. ZFInterfaceOrientationMask orientationMask = [ZFLandscapeRotationManager supportedInterfaceOrientationsForWindow:window];
  131. if (orientationMask != ZFInterfaceOrientationMaskUnknow) {
  132. return (UIInterfaceOrientationMask)orientationMask;
  133. }
  134. HLog(@"处理旋转屏幕:%d",self.supportScreenRotateType)
  135. //显示密码了
  136. if(ksharedAppDelegate.isDidShowPwdType){
  137. return UIInterfaceOrientationMaskPortrait;
  138. }
  139. if(self.supportScreenRotateType){
  140. return UIInterfaceOrientationMaskAllButUpsideDown;
  141. }
  142. if(self.isPlayerScreenLandscapeType){
  143. return UIInterfaceOrientationMaskLandscapeRight;
  144. }
  145. return UIInterfaceOrientationMaskPortrait;
  146. }
  147. - (void)applicationWillTerminate:(UIApplication *)application {
  148. //
  149. [AMediaStream globalDeinitialization];
  150. }
  151. #pragma mark - UISceneSession lifecycle
  152. - (UISceneConfiguration *)application:(UIApplication *)application configurationForConnectingSceneSession:(UISceneSession *)connectingSceneSession options:(UISceneConnectionOptions *)options {
  153. // Called when a new scene session is being created.
  154. // Use this method to select a configuration to create the new scene with.
  155. return [[UISceneConfiguration alloc] initWithName:@"Default Configuration" sessionRole:connectingSceneSession.role];
  156. }
  157. - (void)application:(UIApplication *)application didDiscardSceneSessions:(NSSet<UISceneSession *> *)sceneSessions {
  158. // Called when the user discards a scene session.
  159. // If any sessions were discarded while the application was not running, this will be called shortly after application:didFinishLaunchingWithOptions.
  160. // Use this method to release any resources that were specific to the discarded scenes, as they will not return.
  161. }
  162. - (void)applicationWillEnterForeground:(UIApplication *)application {
  163. // Called as part of the transition from the background to the active state; here you can undo many of the changes made on entering the background.
  164. dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.4 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
  165. NSString *orderNumber = [HWDataManager getStringWithKey:Const_AirpayOrWXorderNum];
  166. if (orderNumber && ![orderNumber isEqualToString:@""])
  167. {
  168. [[NSNotificationCenter defaultCenter] postNotificationName:NotNameAirpayOrWXorderNum object:orderNumber];
  169. }
  170. });
  171. }
  172. - (BOOL)application:(UIApplication *)application handleOpenURL:(NSURL *)url {
  173. return [WXApi handleOpenURL:url delegate:self];
  174. }
  175. - (BOOL)application:(UIApplication *)application openURL:(NSURL *)url sourceApplication:(NSString *)sourceApplication annotation:(id)annotation {
  176. if (YES == [TencentOAuth CanHandleOpenURL:url])
  177. {
  178. return [TencentOAuth HandleOpenURL:url];
  179. }
  180. return [WXApi handleOpenURL:url delegate:self];
  181. }
  182. - (BOOL)application:(UIApplication *)application continueUserActivity:(NSUserActivity *)userActivity restorationHandler:(void (^)(NSArray<id<UIUserActivityRestoring>> * _Nullable))restorationHandler
  183. {
  184. if([userActivity.activityType isEqualToString:NSUserActivityTypeBrowsingWeb]) {
  185. NSURL *url = userActivity.webpageURL;
  186. if(url && [TencentOAuth CanHandleUniversalLink:url]) {
  187. UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"CI UniversalLink" message:url.description delegate:nil cancelButtonTitle:@"ok" otherButtonTitles:nil, nil];
  188. [alertView show];
  189. //[QQApiInterface handleOpenUniversallink:url delegate:(id<QQApiInterfaceDelegate>)[QQApiShareEntry class]];
  190. return [TencentOAuth HandleUniversalLink:url];
  191. }
  192. }
  193. return [WXApi handleOpenUniversalLink:userActivity delegate:self];
  194. }
  195. #pragma mark 监听网络变化
  196. -(void)MonitorNetworkChangesFun
  197. {
  198. [[AFNetworkReachabilityManager sharedManager] setReachabilityStatusChangeBlock:^(AFNetworkReachabilityStatus status) {
  199. HLog(@"Reachability: %@", AFStringFromNetworkReachabilityStatus(status));
  200. mainBlock((^{
  201. //[[iToast makeText:[[NSString alloc] initWithFormat:@"Reachability:%@",AFStringFromNetworkReachabilityStatus(status)] ] show];
  202. [[NSNotificationCenter defaultCenter] postNotificationName:NetWorkChangeNotification object:nil];
  203. }));
  204. //网络切换
  205. [self handelNetWorkChangeFunBy:status];
  206. }];
  207. [[AFNetworkReachabilityManager sharedManager] startMonitoring];
  208. }
  209. #pragma mark 处理网络切换情况
  210. - (void)handelNetWorkChangeFunBy:(AFNetworkReachabilityStatus)status
  211. {
  212. if(status == AFNetworkReachabilityStatusReachableViaWiFi){//检测到起切换wifi 重新ping一下
  213. [[pingManager shareManager] startPingDeviceIpFun];
  214. }
  215. else {//非wifi链接
  216. [pingManager shareManager].isPingOk = NO;
  217. }
  218. //是否备份中 处理切换网络环境 用户无感备份切换
  219. if(([nasBackupsManager shareInstance].curPhotosBackupsTaskMod
  220. && [nasBackupsManager shareInstance].curPhotosBackupsTaskMod.curBackupsState == backupsStateUploading)
  221. || [nasBackupsManager shareInstance].isWifiNeedReBackupsType){
  222. //1.先暂停备份
  223. [[nasBackupsManager shareInstance] suspendBackupsFileFun];
  224. //2.重新开启备份
  225. dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(5 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
  226. [[nasBackupsManager shareInstance] reBackupsFileFun];
  227. });
  228. }
  229. //是否上传中 处理切换网络环境 用户无感上传切换
  230. BOOL needToReUploadTaskType = [nasUploadFileManager shareInstance].needToReUploadTaskType;
  231. if(needToReUploadTaskType){//如果网络层先保存 这里为YES
  232. dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(5 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
  233. [[nasUploadFileManager shareInstance] reUploadFileFunByNetWork];
  234. });
  235. }
  236. else{
  237. //有可能是网络变化先收到通知
  238. if([nasUploadFileManager shareInstance].uploadingArr.count > 0){
  239. [[nasUploadFileManager shareInstance] saveUploadingTaskByNetWorkErrorFun];
  240. //因为不知道是网络变化通知快 还是传输快 这里5秒后重新传
  241. dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(5 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
  242. [[nasUploadFileManager shareInstance] reUploadFileFunByNetWork];
  243. });
  244. }
  245. }
  246. //HLog(@"hxd 1111")
  247. //是否下载中 处理切换网络环境 用户无感下载切换
  248. BOOL needToReDownloadTaskType = [nasDownloadFileManager shareInstance].needToReDownloadTaskType;
  249. if(needToReDownloadTaskType){//如果网络层先保存 这里为YES
  250. //HLog(@"hxd 2222")
  251. dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(5 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
  252. [[nasDownloadFileManager shareInstance] reDownloadloadFileFunByNetWork];
  253. });
  254. }
  255. else{
  256. //HLog(@"hxd 3333")
  257. //有可能是网络变化先收到通知
  258. if([nasDownloadFileManager shareInstance].downLoadFileModelDataArr.count > 0){
  259. //HLog(@"hxd 4444")
  260. [[nasDownloadFileManager shareInstance] saveDownloadloadingTaskByNetWorkErrorFun];
  261. //因为不知道是网络变化通知快 还是传输快 这里5秒后重新传
  262. dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(5 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
  263. [[nasDownloadFileManager shareInstance] reDownloadloadFileFunByNetWork];
  264. });
  265. }
  266. }
  267. }
  268. #pragma mark 多国语言相关
  269. -(void)setLanguagesFun{
  270. NSArray *arLanguages = [[NSUserDefaults standardUserDefaults] objectForKey:@"AppleLanguages"];
  271. NSLog(@"arLanguages:%@",arLanguages);
  272. ///获取设备当前地区的代码和APP语言环境
  273. NSString *languageCode = [NSLocale preferredLanguages][0];
  274. // 获取国际通用国家地区代码(应该和手机本身有关)
  275. NSString *countryCode = [NSString stringWithFormat:@"-%@", [[NSLocale currentLocale] objectForKey:NSLocaleCountryCode]];
  276. // if (languageCode) {
  277. // languageCode = [languageCode stringByReplacingOccurrencesOfString:countryCode withString:@""];
  278. // }
  279. NSLog(@"countryCode:%@ languageCode : %@",countryCode, languageCode);
  280. ///当前APP使用的语言
  281. NSString *preferredLanguage = [[[NSBundle mainBundle] preferredLocalizations] firstObject];
  282. //获取设备当前地区的代码和APP语言环境
  283. NSString *localeIdentifier = [[NSLocale currentLocale] objectForKey:NSLocaleIdentifier];
  284. NSLog(@"preferredLanguage:%@ localeIdentifier : %@",preferredLanguage, localeIdentifier);
  285. //目前支持 中文(简体 繁体) 英文 日语(还没做)
  286. if([languageCode rangeOfString:preferredLanguage].location != NSNotFound){
  287. // if([preferredLanguage rangeOfString:@"ja"].location != NSNotFound){
  288. // //日文现在也显示英文
  289. // [self setDefaultEnglishFun];
  290. // }
  291. // else
  292. {
  293. [DDYLanguageTool ddy_SetLanguage:@"" complete:^(NSError *error) {
  294. // 刷新rootVC等
  295. }];
  296. }
  297. }
  298. else{
  299. [self setDefaultEnglishFun];
  300. }
  301. }
  302. #pragma mark 设置当前显示语言为中文
  303. - (void)setDefaultEnglishFun{
  304. //默认设为英文
  305. [DDYLanguageTool ddy_SetLanguage:@"en" complete:^(NSError *error) {
  306. // 刷新rootVC等
  307. }];
  308. }
  309. - (void)batteryLevelChanged:(NSNotification *)notification {
  310. // 获取当前设备的电池电量
  311. UIDevice *device = notification.object;
  312. float batteryLevel = device.batteryLevel;
  313. // 根据电量级别执行相应的操作
  314. if (batteryLevel < 0.2) {
  315. NSLog(@"Low battery level. Please charge the device.");
  316. } else if (batteryLevel < 0.5) {
  317. NSLog(@"Medium battery level.");
  318. } else {
  319. NSLog(@"High battery level.");
  320. }
  321. }
  322. - (void)handleCrashException:(NSString*)exceptionMessage exceptionCategory:(JJExceptionGuardCategory)exceptionCategory extraInfo:(nullable NSDictionary*)info{
  323. NSMutableString *totalLogstr = [NSMutableString new];
  324. if(exceptionMessage){
  325. [totalLogstr appendString:exceptionMessage];
  326. [totalLogstr appendString:@"\n\n\n"];
  327. }
  328. if(info){
  329. NSData *jsonData = [NSJSONSerialization dataWithJSONObject:info options:info error:nil];
  330. if(jsonData && jsonData.length>0){
  331. NSString *jsonStr = [[NSString alloc] initWithData:jsonData encoding:NSUTF8StringEncoding];
  332. if(jsonStr){
  333. [totalLogstr appendString:jsonStr];
  334. }
  335. }
  336. }
  337. //写日志
  338. if(totalLogstr && totalLogstr.length >0){
  339. [cachesFileManager writeCrashLogsWithMsg:totalLogstr];
  340. }
  341. }
  342. - (NSString*)NASShareFileService
  343. {
  344. //http://transfer.armclouding.com:10012/getFile?path=/sdcard/Download/mmexport1712039794930_4124382.png
  345. if(_NASShareFileService && _NASShareFileService.length >0){
  346. return _NASShareFileService;
  347. }
  348. if(_NASMsgMod){
  349. if([_NASMsgMod.data.domainName rangeOfString:@"http"].location != NSNotFound){
  350. _NASShareFileService = [[NSString alloc] initWithFormat:@"%@:%@/",_NASMsgMod.data.domainName,_NASMsgMod.data.port];
  351. //return [[NSString alloc] initWithFormat:@"%@:%@/",@"http://transfer.armclouding.com",@"10016"];
  352. //return [[NSString alloc] initWithFormat:@"%@:%@/",@"http://transfer.armclouding.com",_NASMsgMod.data.port];
  353. }
  354. else{
  355. _NASShareFileService = [[NSString alloc] initWithFormat:@"http://%@:%@/",_NASMsgMod.data.domainName,_NASMsgMod.data.port];
  356. //return [[NSString alloc] initWithFormat:@"http://%@:%@/",@"transfer.armclouding.com",@"10016"];
  357. //return [[NSString alloc] initWithFormat:@"http://%@:%@/",@"transfer.armclouding.com",_NASMsgMod.data.port];
  358. }
  359. //return [[NSString alloc] initWithFormat:@"%@getFile?path=",_NASMsgMod.data.baseUrl];
  360. //return [[NSString alloc] initWithFormat:@"%@/",_NASMsgMod.data.baseUrl];
  361. }
  362. return _NASShareFileService;
  363. }
  364. - (NSString*)NASFileByBoxService
  365. {
  366. //内网情况
  367. if([pingManager shareManager].isPingOk){
  368. if(NASFileByBoxServiceByPingok && NASFileByBoxServiceByPingok.length >0){
  369. return NASFileByBoxServiceByPingok;
  370. }
  371. NASFileByBoxServiceByPingok = [NSString stringWithFormat:@"http://%@:9888/",ksharedAppDelegate.DeviceThirdIdMod.data.ip];
  372. return NASFileByBoxServiceByPingok;
  373. }
  374. //外网情况
  375. if(_NASMsgMod){
  376. if(NASFileByBoxServiceByPingNot && NASFileByBoxServiceByPingNot.length >0){
  377. return NASFileByBoxServiceByPingNot;
  378. }
  379. if([_NASMsgMod.data.domainName rangeOfString:@"http"].location != NSNotFound){
  380. NASFileByBoxServiceByPingNot = [[NSString alloc] initWithFormat:@"%@:%@/",_NASMsgMod.data.domainName,_NASMsgMod.data.port];
  381. }
  382. else{
  383. NASFileByBoxServiceByPingNot = [[NSString alloc] initWithFormat:@"http://%@:%@/",_NASMsgMod.data.domainName,_NASMsgMod.data.port];
  384. }
  385. return NASFileByBoxServiceByPingNot;
  386. }
  387. return @"";
  388. }
  389. #pragma mark 更换设备 重新设置地址
  390. - (void)resetBoxNetUrlFun
  391. {
  392. _NASShareFileService = nil;
  393. NASFileByBoxServiceByPingok = nil;
  394. NASFileByBoxServiceByPingNot = nil;
  395. }
  396. - (void)imageLoadingSettings {
  397. [SDImageCache sharedImageCache].config.maxDiskAge = 3600 * 24 * 7;
  398. [SDImageCache sharedImageCache].config.maxMemoryCount = 1024 * 1024 * 20;
  399. [SDImageCache sharedImageCache].config.shouldCacheImagesInMemory = YES;
  400. //[SDImageCache sharedImageCache].config.shouldDecompressImages = NO;
  401. //[SDWebImageDownloader sharedDownloader].shouldDecompressImages = NO;
  402. [SDImageCache sharedImageCache].config.diskCacheReadingOptions = NSDataReadingMappedIfSafe;
  403. }
  404. @end