首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >在运行时创建UIApplicationDelegate类

在运行时创建UIApplicationDelegate类
EN

Stack Overflow用户
提问于 2016-10-09 19:31:07
回答 1查看 310关注 0票数 2

使用目标C运行时,我试图在运行时为我的iOS应用程序创建一个iOS。这只是为了研究的目的,我没有打算运送这个。

到目前为止,我的步骤是:

  • 在运行时创建一个名为AppDelegate的类。
  • 添加一个名为_window的实例变量。
  • 添加一个名为window的属性,该属性使用实例变量作为支持变量,并使用两个C函数作为getter和setter。
  • 为选择器application:didFinishLaunchingWithOptions:添加一个方法,并在C中使用返回YES的实现。此时,该类实现了UIApplicationDelegate协议。
  • 将类注册到运行时。

然而,当我在连接的iPhone上启动程序时,屏幕仍然是黑色的,尽管程序实际上没有崩溃。调试器显示调用了我的实现。在文档之后,操作系统首先检查我的window属性是否为nil (它是),然后创建一个UIWindow本身,并使用delegate将UIWindow分配给我的委托实例。

当我访问这个窗口时,它似乎是完全可用的:它具有通常的界限(NSRect: Height = 667;Width = 375;X= 0;Y= 0;),并且我的自定义ViewController被初始化,viewDidLoad被调用。

有人能帮我找出为什么屏幕是黑色的吗?

[self.window makeKeyAndVisible]的调用不起作用,屏幕保持黑色。如果我自己在application:didFinishLaunchingWithOptions:中创建了一个窗口,并分配了前一个UIWindow的rootViewController,那么这是可行的。所以一定是我的代表中的UIWindow被打破了。

下面是我使用的代码:

代码语言:javascript
复制
#import <UIKit/UIKit.h>
#import <objc/runtime.h>
#import <stdio.h>


id getter(id self, SEL _cmd) {
    Ivar ivar = class_getInstanceVariable(objc_getClass("AppDelegate"), "_window");
    id var = object_getIvar(self, ivar);
    printf("is nil: %s\n", var == nil ? "true" : "false");
    if (var != nil) {
        printf("%s\n", [[NSString stringWithFormat:@"%@", CGRectCreateDictionaryRepresentation(((UIWindow *) var).bounds)] UTF8String]);
    }
    return object_getIvar(self, ivar);
}

void setter(id self, SEL _cmd, id new) {
    printf("setter...\n");
    Ivar ivar = class_getInstanceVariable(objc_getClass("AppDelegate"), "_window");
    object_setIvar(self, ivar, new);
}

BOOL didFinishLaunching(id self, SEL _cmd, id launchOptions) {
    printf("didFinishLaunching called\n");
    return YES;
}


int main(int argc, char * argv[]) {
    @autoreleasepool {
        Class delegate = objc_allocateClassPair([NSObject class], "AppDelegate", 0);
        class_addIvar(delegate, "_window", sizeof(UIWindow *), rint(log2(sizeof(UIWindow *))), @encode(UIWindow *));

        objc_property_attribute_t type = { "T", "@\"UIWindow\"" };
        objc_property_attribute_t strength = { "&", "" };
        objc_property_attribute_t atomic = { "N", "" };
        objc_property_attribute_t backingVar = { "V", "_window" };
        objc_property_attribute_t attrs[] = { type, strength, atomic, backingVar };
        class_addProperty(delegate, "window", attrs, 4);

        class_addMethod(delegate, @selector(window), (IMP) getter, "@@:");

        class_addMethod(delegate, @selector(setWindow:), (IMP) setter, "v@:@");

        class_addMethod(delegate, @selector(application:didFinishLaunchingWithOptions:),
                        (IMP) didFinishLaunching, "B@:@");
        objc_registerClassPair(delegate);

        return UIApplicationMain(argc, argv, nil, @"AppDelegate");
    }
}

这是标准输出:

代码语言:javascript
复制
is nil: true
setter...
didFinishLaunching called
is nil: false
{
    Height = 667;
    Width = 375;
    X = 0;
    Y = 0;
}
is nil: false
{
    Height = 667;
    Width = 375;
    X = 0;
    Y = 0;
}
ViewController.viewDidLoad() called
is nil: false
{
    Height = 667;
    Width = 375;
    X = 0;
    Y = 0;
}
EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2016-10-10 21:48:23

发现了问题:弧形正在释放我的UIWindow。

将设置者更改为

代码语言:javascript
复制
void setter(id self, SEL _cmd, id new) {
    Ivar ivar = class_getInstanceVariable(object_getClass(self), "_window");
    id old = object_getIvar(self, ivar);
    if (![old isEqual: new]) {
        if(old != nil) {
            objc_msgSend(old, sel_getUid("release"));
        }
        object_setIvar(self, ivar, new);
        objc_msgSend(new, sel_getUid("retain"));
    }
}

解决这个问题。

票数 2
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/39947685

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档