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

相关文章
|
26天前
|
Java Android开发 Swift
安卓与iOS开发对比:平台选择对项目成功的影响
【10月更文挑战第4天】在移动应用开发的世界中,选择合适的平台是至关重要的。本文将深入探讨安卓和iOS两大主流平台的开发环境、用户基础、市场份额和开发成本等方面的差异,并分析这些差异如何影响项目的最终成果。通过比较这两个平台的优势与挑战,开发者可以更好地决定哪个平台更适合他们的项目需求。
97 1
|
1月前
|
设计模式 安全 Swift
探索iOS开发:打造你的第一个天气应用
【9月更文挑战第36天】在这篇文章中,我们将一起踏上iOS开发的旅程,从零开始构建一个简单的天气应用。文章将通过通俗易懂的语言,引导你理解iOS开发的基本概念,掌握Swift语言的核心语法,并逐步实现一个具有实际功能的天气应用。我们将遵循“学中做,做中学”的原则,让理论知识和实践操作紧密结合,确保学习过程既高效又有趣。无论你是编程新手还是希望拓展技能的开发者,这篇文章都将为你打开一扇通往iOS开发世界的大门。
|
1月前
|
搜索推荐 IDE API
打造个性化天气应用:iOS开发之旅
【9月更文挑战第35天】在这篇文章中,我们将一起踏上iOS开发的旅程,通过创建一个个性化的天气应用来探索Swift编程语言的魅力和iOS平台的强大功能。无论你是编程新手还是希望扩展你的技能集,这个项目都将为你提供实战经验,帮助你理解从构思到实现一个应用的全过程。让我们开始吧,构建你自己的天气应用,探索更多可能!
60 1
|
2天前
|
设计模式 前端开发 Swift
探索iOS开发:从初级到高级的旅程
【10月更文挑战第28天】在这篇技术性文章中,我们将一起踏上一段探索iOS开发的旅程。无论你是刚入门的新手,还是希望提升技能的开发者,这篇文章都将为你提供宝贵的指导和灵感。我们将从基础概念开始,逐步深入到高级主题,如设计模式、性能优化等。通过阅读这篇文章,你将获得一个清晰的学习路径,帮助你在iOS开发领域不断成长。
23 2
|
7天前
|
安全 API Swift
探索iOS开发中的Swift语言之美
【10月更文挑战第23天】在数字时代的浪潮中,iOS开发如同一艘航船,而Swift语言则是推动这艘船前进的风帆。本文将带你领略Swift的独特魅力,从语法到设计哲学,再到实际应用案例,我们将一步步深入这个现代编程语言的世界。你将发现,Swift不仅仅是一种编程语言,它是苹果生态系统中的一个创新工具,它让iOS开发变得更加高效、安全和有趣。让我们一起启航,探索Swift的奥秘,感受编程的乐趣。
|
9天前
|
Swift iOS开发 开发者
探索iOS开发中的SwiftUI框架
【10月更文挑战第21天】在苹果生态系统中,SwiftUI的引入无疑为iOS应用开发带来了革命性的变化。本文将通过深入浅出的方式,带领读者了解SwiftUI的基本概念、核心优势以及如何在实际项目中运用这一框架。我们将从一个简单的例子开始,逐步深入到更复杂的应用场景,让初学者能够快速上手,同时也为有经验的开发者提供一些深度使用的技巧和策略。
34 1
|
27天前
|
移动开发 前端开发 Swift
iOS 最好的应用程序开发编程语言竟然是这7种
iOS 最好的应用程序开发编程语言竟然是这7种
72 8
|
25天前
|
Android开发 Swift iOS开发
探索安卓与iOS开发的差异:从代码到用户体验
【10月更文挑战第5天】在移动应用开发的广阔天地中,安卓和iOS两大平台各占半壁江山。它们在技术架构、开发环境及用户体验上有着根本的不同。本文通过比较这两种平台的开发过程,揭示背后的设计理念和技术选择如何影响最终产品。我们将深入探讨各自平台的代码示例,理解开发者面临的挑战,以及这些差异如何塑造用户的日常体验。
|
29天前
|
安全 Swift iOS开发
探索iOS开发中的Swift语言之美
在数字时代的浪潮中,移动应用已成为日常生活的延伸。本文将深入探讨iOS平台上的Swift编程语言,揭示其背后的设计哲学、语法特性以及如何利用Swift进行高效开发。我们将通过实际代码示例,展示Swift语言的强大功能和优雅简洁的编程风格,引导读者理解并运用Swift解决实际问题。
|
1月前
|
开发框架 移动开发 Android开发
安卓与iOS开发中的跨平台解决方案:Flutter入门
【9月更文挑战第30天】在移动应用开发的广阔舞台上,安卓和iOS两大操作系统各自占据半壁江山。开发者们常常面临着选择:是专注于单一平台深耕细作,还是寻找一种能够横跨两大系统的开发方案?Flutter,作为一种新兴的跨平台UI工具包,正以其现代、响应式的特点赢得开发者的青睐。本文将带你一探究竟,从Flutter的基础概念到实战应用,深入浅出地介绍这一技术的魅力所在。
71 7