草庐IT

iOS开发之自定义Window

LazyLoad 2023-03-28 原文

前言

这篇文章主要记录在我在开发中针对 UIWindow 的使用。

遇到的问题

通常境况下,我在新建一个新的 iOS 项目后,每次都会删除 main.storyboard 这个文件。然后自己在 AppDelegate 中自己来创建一个 window 对象。大致就是下面这个样子的:

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
    
    self.window = [[UIWindow alloc] initWithFrame:[UIScreen mainScreen].bounds];
    
    self.window.backgroundColor = [UIColor blueColor];
    
    self.window.rootViewController = [ViewController new];
    
    [self.window makeKeyAndVisible];
    
    return YES;
}

最近在项目中遇到一个场景:

App 启动后,此时需要 App 进行强制更新。强制更新,就要求强制更新的按钮在最顶层,我发现之前代码是通过 [UIApplication sharedApplication].delegate.window 获取到 window 对象,并在此之上添加强制更新的视图作为它的子视图。貌似好像没有什么问题,其实不然。因为这个强制更新的视图不一定是在最顶层的。因为 Modal 出来的 Controller 会把它盖住。不信你就试试,这样一来可以进行别的操作算哪门子的强制更新?

解决方案

既然 KeyWindow 解决不了这个问题,我的解决方案是通过自定义 window,并设置 window 的级别,这样他就在最顶层了。而自己创建一个 window 对象是有一些小细节的。还是通过代码具体看一下:

我打算怎么设计

  • 自定义 YMUpdateView 继承自 UIView,提供一个 show 接口
  • 重写 \- (**instancetype**)initWithFrame:(CGRect)frame 方法,自定义 UI 视图,主要就是自定义 window 对象。window 里面的子视图,可以根据自己的业务和 UI 设计稿进行自己定制

我的实现

部分代码如下:

#import "YMUpdateView.h"

@interface UpdateView ()

@property (nonatomic, strong) UIWindow *window;

@end

@implementation UpdateView

- (instancetype)initWithFrame:(CGRect)frame {
    if (self = [super initWithFrame:frame]) {
        
        
        _window = [[UIWindow alloc] initWithFrame:[UIScreen mainScreen].bounds];
        _window.windowLevel = UIWindowLevelStatusBar;
        _window.rootViewController = [[UIViewController alloc] init];
        _window.backgroundColor = [[UIColor blackColor] colorWithAlphaComponent:0.6];
        
        
    }
    return self;
}


- (void)show {
    _window.hidden = NO;
}

@end

  • 核心代码就是 \- (**instancetype**)initWithFrame:(CGRect)frame 中自己创建一个 window 对象。更好一点可以封装一个方法专门来创建 UI 视图。在 layoutSubviews 方法中进行布局。
  • 注意一点就是,window 对象必须是被强引用的
  • windowLevel 设置为 UIWindowLevelStatusBar 级别,它和状态栏在一个级别,一定是在最顶层的
  • 在使用的时候,也要用 strong 修饰的属性,进行强引用
@interface ViewController ()
@property (nonatomic, strong) UpdateView *uv;
@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    self.uv = [[UpdateView alloc] init];
    [self.uv show];
    
    
    // 模拟出现 modal Controller 盖住的情况
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        TestViewController *vc = [[TestViewController alloc] init];

        vc.modalPresentationStyle = UIModalPresentationFullScreen;
        [self presentViewController:vc animated:YES completion:nil];
    });

} /* viewDidLoad */

结束语

东西不难,但在工作中还比较实用。这只是一个简单的场景。我们经常在一些 App 中见到悬浮球的功能,我觉得也可以通过自定义 window 对象来实现,需要处理的是手势和边界的计算。

有关iOS开发之自定义Window的更多相关文章

  1. ruby - Facter::Util::Uptime:Module 的未定义方法 get_uptime (NoMethodError) - 2

    我正在尝试设置一个puppet节点,但ruby​​gems似乎不正常。如果我通过它自己的二进制文件(/usr/lib/ruby/gems/1.8/gems/facter-1.5.8/bin/facter)在cli上运行facter,它工作正常,但如果我通过由ruby​​gems(/usr/bin/facter)安装的二进制文件,它抛出:/usr/lib/ruby/1.8/facter/uptime.rb:11:undefinedmethod`get_uptime'forFacter::Util::Uptime:Module(NoMethodError)from/usr/lib/ruby

  2. ruby-on-rails - Rails 3.2.1 中 ActionMailer 中的未定义方法 'default_content_type=' - 2

    我在我的项目中添加了一个系统来重置用户密码并通过电子邮件将密码发送给他,以防他忘记密码。昨天它运行良好(当我实现它时)。当我今天尝试启动服务器时,出现以下错误。=>BootingWEBrick=>Rails3.2.1applicationstartingindevelopmentonhttp://0.0.0.0:3000=>Callwith-dtodetach=>Ctrl-CtoshutdownserverExiting/Users/vinayshenoy/.rvm/gems/ruby-1.9.3-p0/gems/actionmailer-3.2.1/lib/action_mailer

  3. ruby - 使用 C 扩展开发 ruby​​gem 时,如何使用 Rspec 在本地进行测试? - 2

    我正在编写一个包含C扩展的gem。通常当我写一个gem时,我会遵循TDD的过程,我会写一个失败的规范,然后处理代码直到它通过,等等......在“ext/mygem/mygem.c”中我的C扩展和在gemspec的“扩展”中配置的有效extconf.rb,如何运行我的规范并仍然加载我的C扩展?当我更改C代码时,我需要采取哪些步骤来重新编译代码?这可能是个愚蠢的问题,但是从我的gem的开发源代码树中输入“bundleinstall”不会构建任何native扩展。当我手动运行rubyext/mygem/extconf.rb时,我确实得到了一个Makefile(在整个项目的根目录中),然后当

  4. ruby-on-rails - form_for 中不在模型中的自定义字段 - 2

    我想向我的Controller传递一个参数,它是一个简单的复选框,但我不知道如何在模型的form_for中引入它,这是我的观点:{:id=>'go_finance'}do|f|%>Transferirde:para:Entrada:"input",:placeholder=>"Quantofoiganho?"%>Saída:"output",:placeholder=>"Quantofoigasto?"%>Nota:我想做一个额外的复选框,但我该怎么做,模型中没有一个对象,而是一个要检查的对象,以便在Controller中创建一个ifelse,如果没有检查,请帮助我,非常感谢,谢谢

  5. ruby - 主要 :Object when running build from sublime 的未定义方法 `require_relative' - 2

    我已经从我的命令行中获得了一切,所以我可以运行rubymyfile并且它可以正常工作。但是当我尝试从sublime中运行它时,我得到了undefinedmethod`require_relative'formain:Object有人知道我的sublime设置中缺少什么吗?我正在使用OSX并安装了rvm。 最佳答案 或者,您可以只使用“require”,它应该可以正常工作。我认为“require_relative”仅适用于ruby​​1.9+ 关于ruby-主要:Objectwhenrun

  6. Ruby Sinatra 配置用于生产和开发 - 2

    我已经在Sinatra上创建了应用程序,它代表了一个简单的API。我想在生产和开发上进行部署。我想在部署时选择,是开发还是生产,一些方法的逻辑应该改变,这取决于部署类型。是否有任何想法,如何完成以及解决此问题的一些示例。例子:我有代码get'/api/test'doreturn"Itisdev"end但是在部署到生产环境之后我想在运行/api/test之后看到ItisPROD如何实现? 最佳答案 根据SinatraDocumentation:EnvironmentscanbesetthroughtheRACK_ENVenvironm

  7. ruby - 在 Ruby 中有条件地定义函数 - 2

    我有一些代码在几个不同的位置之一运行:作为具有调试输出的命令行工具,作为不接受任何输出的更大程序的一部分,以及在Rails环境中。有时我需要根据代码的位置对代码进行细微的更改,我意识到以下样式似乎可行:print"Testingnestedfunctionsdefined\n"CLI=trueifCLIdeftest_printprint"CommandLineVersion\n"endelsedeftest_printprint"ReleaseVersion\n"endendtest_print()这导致:TestingnestedfunctionsdefinedCommandLin

  8. ruby - 定义方法参数的条件 - 2

    我有一个只接受一个参数的方法:defmy_method(number)end如果使用number调用方法,我该如何引发错误??通常,我如何定义方法参数的条件?比如我想在调用的时候报错:my_method(1) 最佳答案 您可以添加guard在函数的开头,如果参数无效则引发异常。例如:defmy_method(number)failArgumentError,"Inputshouldbegreaterthanorequalto2"ifnumbereputse.messageend#=>Inputshouldbegreaterthano

  9. ruby - 如何验证 IO.copy_stream 是否成功 - 2

    这里有一个很好的答案解释了如何在Ruby中下载文件而不将其加载到内存中:https://stackoverflow.com/a/29743394/4852737require'open-uri'download=open('http://example.com/image.png')IO.copy_stream(download,'~/image.png')我如何验证下载文件的IO.copy_stream调用是否真的成功——这意味着下载的文件与我打算下载的文件完全相同,而不是下载一半的损坏文件?documentation说IO.copy_stream返回它复制的字节数,但是当我还没有下

  10. ruby - 如何在 Grape 中定义哈希数组? - 2

    我使用Ember作为我的前端和GrapeAPI来为我的API提供服务。前端发送类似:{"service"=>{"name"=>"Name","duration"=>"30","user"=>nil,"organization"=>"org","category"=>nil,"description"=>"description","disabled"=>true,"color"=>nil,"availabilities"=>[{"day"=>"Saturday","enabled"=>false,"timeSlots"=>[{"startAt"=>"09:00AM","endAt"=>

随机推荐