iOS开发实战 - 完美解决UIScrollView嵌套滑动手势冲突

简介: 我们应该都有用过这个功能,你的朋友微信给你分享了一个淘宝里面的商品链接,然后当你复制这个链接打开淘宝APP的时候,就会弹出一个弹窗,像这样:example.PNG这个功能想必大家都挺熟悉,受这个启发我们产品也想在我们APP上添加这样一个功能,与这个不一样的是,当我们复制一段网址的时候打开我们的APP会弹出框填一些信息后上传到我们的“资源库”。

我们应该都有用过这个功能,你的朋友微信给你分享了一个淘宝里面的商品链接,然后当你复制这个链接打开淘宝APP的时候,就会弹出一个弹窗,像这样:

example.PNG

这个功能想必大家都挺熟悉,受这个启发我们产品也想在我们APP上添加这样一个功能,与这个不一样的是,当我们复制一段网址的时候打开我们的APP会弹出框填一些信息后上传到我们的“资源库”。大体功能就这样,所以记录一下实现的过程。


一、弹窗视图功能

.h中:两个信号一个是确定信号一个是取消信号

两个方法,一个显示一个隐藏方法

1
2
3
4
5
@property (nonatomic, strong) RACSubject *uploadSureSignal; //确定上传信号
@property (nonatomic, strong) RACSubject *hideSucSignal; //隐藏
 
- ( void )show;
- ( void )hide;

.m中:主要是两个textview,还有涉及到在keywindow上,IQKeyboard的一些操作

1
2
3
@property (nonatomic, assign) CGFloat keyboardHeight; //键盘高度
@property (nonatomic, strong) CustomUITextView *nameTV;
@property (nonatomic, strong) CustomUITextView *desTV;

因为发现IQKeyboard在这个弹出界面有问题,所以在显示这个界面的时候,将IQKeyboard禁用取之使用系统的keyboard监听方法

在(void)show方法中:

1
2
3
4
5
6
7
8
-( void )show {
//键盘通知
     NSNotificationCenter *defaultCenter = [NSNotificationCenter defaultCenter];
     
     [defaultCenter addObserver:self selector:@selector(keyboardWillShowOrHide:) name:UIKeyboardWillShowNotification object:nil];
     
     [defaultCenter addObserver:self selector:@selector(keyboardWillShowOrHide:) name:UIKeyboardWillHideNotification object:nil];
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
//监听方法
- ( void )keyboardWillShowOrHide:(NSNotification *)notification {
     //获取通知名
     NSString *notificationName = notification.name;
     //获取通知内容
     NSDictionary *keyboardInfo = notification.userInfo;
     //键盘弹出时,让画面整体稍稍上移,并伴随动画
     //键盘回收时反之
     CGRect keyboardFrame = [notification.userInfo[UIKeyboardFrameEndUserInfoKey] CGRectValue];
     CGFloat height = keyboardFrame.size.height;
     self.keyboardHeight = height;
     //动画结束后self.view的frame值
     CGRect selfViewFrame = self.bgView.frame;
     //通过通知名字判断弹出还是回收
     if  ([notificationName isEqualToString:UIKeyboardWillShowNotification]) {
         selfViewFrame.origin.y = SCREEN_HEIGHT - PANELHEIGHT - height;
     else  {
         selfViewFrame.origin.y = SCREEN_HEIGHT - PANELHEIGHT;
     }
     
     //取出动画时长
     NSTimeInterval duration = [keyboardInfo[UIKeyboardAnimationDurationUserInfoKey] doubleValue];
     
     //使用动画更改self.view.frame
     [UIView animateWithDuration:duration animations:^{
         //这里填入一些view的最终状态属性设置,即会自动产生过渡动画
         self.bgView.frame = selfViewFrame;
     }];
     
}

同时在show方法中显示keyWindow,进而改变界面的frame进行显示

1
2
3
4
5
6
7
8
9
10
- ( void )show {
UIWindow *keyWindow = [UIApplication sharedApplication].keyWindow;
     [keyWindow addSubview:self];
     CGRect frame = self.bgView.frame;
     if  (frame.origin.y == SCREEN_HEIGHT) {
         frame.origin.y = SCREEN_HEIGHT - PANELHEIGHT;
         [UIView animateWithDuration: 0.4  animations:^{
             self.bgView.frame = frame;
         }];
     }

hide方法这里要考虑到键盘弹出后将self.bgView向上提高后frame的变化。

1
2
3
4
5
6
7
8
9
10
11
12
13
CGRect selfFrame = self.bgView.frame;
     if  (selfFrame.origin.y == SCREEN_HEIGHT - PANELHEIGHT || selfFrame.origin.y == SCREEN_HEIGHT - PANELHEIGHT - self.keyboardHeight) {
         [self resignFirstResponder];
         selfFrame.origin.y = SCREEN_HEIGHT;
         [UIView animateWithDuration: 0.4  animations:^{
             self.bgView.frame = selfFrame;
         } completion:^(BOOL finished) {
             [IQKeyboardManager sharedManager].enable = YES;
             [[NSNotificationCenter defaultCenter] removeObserver:self];
//            [self.hideSucSignal sendNext:nil];
             [self removeFromSuperview];
         }];
     }

delegate中的操作

这里首先要弄懂APPdelegate中的这几个代理方法的意思:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
//App已经启动
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
     // Override point for customization after application launch.
     return  YES;
}
 
//App挂起状态
- ( void )applicationWillResignActive:(UIApplication *)application {
     // Sent when the application is about to move from active to inactive state. This can occur for certain types of temporary interruptions (such as an incoming phone call or SMS message) or when the user quits the application and it begins the transition to the background state.
     // Use this method to pause ongoing tasks, disable timers, and invalidate graphics rendering callbacks. Games should use this method to pause the game.
}
 
//APP进入后台
- ( void )applicationDidEnterBackground:(UIApplication *)application {
     // Use this method to release shared resources, save user data, invalidate timers, and store enough application state information to restore your application to its current state in case it is terminated later.
     // If your application supports background execution, this method is called instead of applicationWillTerminate: when the user quits.
}
 
//APP将重新回到前台
- ( void )applicationWillEnterForeground:(UIApplication *)application {
     // 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.
}
 
//APP进入活跃状态
- ( void )applicationDidBecomeActive:(UIApplication *)application {
     // Restart any tasks that were paused (or not yet started) while the application was inactive. If the application was previously in the background, optionally refresh the user interface.
}
 
//系统时间发生改变时执行
- ( void )applicationWillTerminate:(UIApplication *)application {
     // Called when the application is about to terminate. Save data if appropriate. See also applicationDidEnterBackground:.
}

在上面的这些代理方法中,我们需要用到的是  applicationDidBecomeActive方法。在这个方法中我们去检查系统的粘贴板UIPasteboard *pasteboard = [UIPasteboard generalPasteboard];

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
if  (pasteboard.string) {
             NSLog(@ "string:%@" , pasteboard.string);
             NSString *urlStr = pasteboard.string;
             if  ([urlStr hasPrefix:@ "https://" ] || [urlStr hasPrefix:@ "http://" ]) {
//如果粘贴板中的字符串包含https或http字段,我们去检查当前的控制器 如果当前的控制器是我们弹出做操作的控制器的话 isPopVC = NO;
                 BOOL isPopVC = NO;
                 UIViewController * Rootvc = self.window.rootViewController;
                 if  ([Rootvc isKindOfClass:[UINavigationController  class ]]) {
                     UINavigationController * nav = (UINavigationController *)Rootvc;
                     UIViewController * v = [nav.viewControllers lastObject];
                     if  ([v isKindOfClass:[UploadResCofingVC  class ]]) {
                         isPopVC = YES;
                     }
                 }
                 //如果popView == nil 并且isPopVC == NO 弹出popView弹窗视图 进行操作
                 if  (!self.popView && !isPopVC) {
                     UploadResourcesPopupView *popView = [UploadResourcesPopupView  new ];
                     [popView show];
                     self.popView = popView;
                     [self.popView.hideSucSignal subscribeNext:^(id x) {
                         @strongify(self);
                         self.popView = nil;
                     }];
                 }
             }
         }
     }

总结

以上大体就是实现这个功能的基本思路,细节方面因项目而异了,比如我们需要判断当前用户的角色,当前用户是否登录,对弹窗视图后续的一些操作。当然并不完美,欢迎批评指正。


转自:http://www.cocoachina.com/ios/20180508/23307.html

相关文章
|
2月前
|
开发框架 前端开发 Android开发
安卓与iOS开发中的跨平台策略
在移动应用开发的战场上,安卓和iOS两大阵营各据一方。随着技术的演进,跨平台开发框架成为开发者的新宠,旨在实现一次编码、多平台部署的梦想。本文将探讨跨平台开发的优势与挑战,并分享实用的开发技巧,帮助开发者在安卓和iOS的世界中游刃有余。
|
25天前
|
iOS开发 开发者 MacOS
深入探索iOS开发中的SwiftUI框架
【10月更文挑战第21天】 本文将带领读者深入了解Apple最新推出的SwiftUI框架,这一革命性的用户界面构建工具为iOS开发者提供了一种声明式、高效且直观的方式来创建复杂的用户界面。通过分析SwiftUI的核心概念、主要特性以及在实际项目中的应用示例,我们将展示如何利用SwiftUI简化UI代码,提高开发效率,并保持应用程序的高性能和响应性。无论你是iOS开发的新手还是有经验的开发者,本文都将为你提供宝贵的见解和实用的指导。
118 66
|
11天前
|
存储 监控 API
app开发之安卓Android+苹果ios打包所有权限对应解释列表【长期更新】-以及默认打包自动添加权限列表和简化后的基本打包权限列表以uniapp为例-优雅草央千澈
app开发之安卓Android+苹果ios打包所有权限对应解释列表【长期更新】-以及默认打包自动添加权限列表和简化后的基本打包权限列表以uniapp为例-优雅草央千澈
|
1月前
|
开发框架 Android开发 iOS开发
安卓与iOS开发中的跨平台策略:一次编码,多平台部署
在移动应用开发的广阔天地中,安卓和iOS两大阵营各占一方。随着技术的发展,跨平台开发框架应运而生,它们承诺着“一次编码,到处运行”的便捷。本文将深入探讨跨平台开发的现状、挑战以及未来趋势,同时通过代码示例揭示跨平台工具的实际运用。
107 3
|
1月前
|
Java 调度 Android开发
安卓与iOS开发中的线程管理差异解析
在移动应用开发的广阔天地中,安卓和iOS两大平台各自拥有独特的魅力。如同东西方文化的差异,它们在处理多线程任务时也展现出不同的哲学。本文将带你穿梭于这两个平台之间,比较它们在线程管理上的核心理念、实现方式及性能考量,助你成为跨平台的编程高手。
|
2月前
|
存储 前端开发 Swift
探索iOS开发:从新手到专家的旅程
本文将带您领略iOS开发的奇妙之旅,从基础概念的理解到高级技巧的掌握,逐步深入iOS的世界。文章不仅分享技术知识,还鼓励读者在编程之路上保持好奇心和创新精神,实现个人成长与技术突破。
|
2月前
|
安全 IDE Swift
探索iOS开发之旅:从初学者到专家
在这篇文章中,我们将一起踏上iOS开发的旅程,从基础概念的理解到深入掌握核心技术。无论你是编程新手还是希望提升技能的开发者,这里都有你需要的指南和启示。我们将通过实际案例和代码示例,展示如何构建一个功能齐全的iOS应用。准备好了吗?让我们一起开始吧!
|
2月前
|
安全 Swift iOS开发
Swift 与 UIKit 在 iOS 应用界面开发中的关键技术和实践方法
本文深入探讨了 Swift 与 UIKit 在 iOS 应用界面开发中的关键技术和实践方法。Swift 以其简洁、高效和类型安全的特点,结合 UIKit 丰富的组件和功能,为开发者提供了强大的工具。文章从 Swift 的语法优势、类型安全、编程模型以及与 UIKit 的集成,到 UIKit 的主要组件和功能,再到构建界面的实践技巧和实际案例分析,全面介绍了如何利用这些技术创建高质量的用户界面。
40 2
|
2月前
|
vr&ar Android开发 iOS开发
安卓与iOS开发中的用户界面设计原则
【10月更文挑战第41天】探索移动应用开发的精髓,本文将深入分析安卓和iOS平台上用户界面设计的核心原则。通过比较两大操作系统的设计哲学,我们将揭示如何打造直观、易用且美观的应用程序界面。无论你是初学者还是资深开发者,这篇文章都将为你提供宝贵的见解和实用的技巧,帮助你在竞争激烈的应用市场中脱颖而出。
|
2月前
|
设计模式 Swift iOS开发
探索iOS开发:从基础到高级,打造你的第一款App
【10月更文挑战第40天】在这个数字时代,掌握移动应用开发已成为许多技术爱好者的梦想。本文将带你走进iOS开发的世界,从最基础的概念出发,逐步深入到高级功能实现,最终指导你完成自己的第一款App。无论你是编程新手还是有志于扩展技能的开发者,这篇文章都将为你提供一条清晰的学习路径。让我们一起开始这段旅程吧!