1

我正在将一个 Objective-C/Cocoa 的小库移植到 Python/PyObjC 中。而且我对实施方式感到困惑。

原始代码如下所示:

@implementation HUDWindow
- (id)initWithContentRect:(NSRect)contentRect 
                styleMask:(unsigned int)styleMask 
                  backing:(NSBackingStoreType)bufferingType 
                    defer:(BOOL)flag 
{
    if (self = [super initWithContentRect:contentRect 
                                styleMask:NSBorderlessWindowMask 
                                  backing:bufferingType 
                                    defer:flag]) {
        // initialize code here
             :
             :
             :
             :
             :
             :
        return self;
    }
}

在头文件中说HUDWindow类继承了NSPanel。据我阅读代码,初始化是这样的:

  1. HUDWindow 的 initContentRect 覆盖 NSPanel 的 initContentRect。
  2. 在 HUDWindow 的 initContentRect 内部,NSPanel 的 initContentRect 被调用并得到结果为“self”。

首先,我这样写python代码。这是一个字一个字的翻译:

class HUDWindow(NSPanel):
    def initWithContentRect_styleMask_backing_defer_(self, rect, style, buf, flag):
        self = NSPanel.initWithContentRect_styleMask_backing_defer_(
            rect, NSBorderlessWindowMask, buf, flag)
        # initialize code here
             :
             :
             :
             :
             :
             :
        return self

它不起作用,因为self它只是NSPanel. 我不能调用从原始 obj-c 代码移植的任何令人上瘾的函数。

所以我像这样更改了我的代码:

class HUDWindow(NSPanel):
    def initWithContentRect_styleMask_backing_defer_(self, rect, style, buf, flag):
        self.window = NSPanel.initWithContentRect_styleMask_backing_defer_(
            rect, NSBorderlessWindowMask, buf, flag)
        # initialize code here
             :
             :
             :
             :
             :
             :
        return self.window

它工作得更好一点。至少,我可以在这个初始化函数中调用其他函数。但是在初始化函数之外,我不能调用从原始代码移植的其他函数。

所以我尝试这样:

class HUDWindow(NSPanel):
    def initWithContentRect_styleMask_backing_defer_(self, rect, style, buf, flag):
        self.window = NSPanel.initWithContentRect_styleMask_backing_defer_(
            rect, NSBorderlessWindowMask, buf, flag)
        # initialize code here
             :
             :
             :
             :
             :
             :
         self.window.setTitle_ = self.setTitle_
         return self.window

它没有用。创建实例后我无法覆盖或添加函数。

现在我觉得我做错了。是否有任何常见的方法来继承像 NSPanel 或 NPWindow 这样的类?

4

1 回答 1

1

您的第一个解决方案是正确的。在第二种情况下,您需要使用NSProxy或编写自己的显式委托,这很痛苦(第三种方法不是这样做的——您可以通过 swizzling 做类似的事情,但不要这样做)。

但是,有一些问题。

首先,这段代码:

self = NSPanel.initWithContentRect_styleMask_backing_defer_(rect, NSBorderlessWindowMask, buf, flag)

… 相当于这个 ObjC 代码:

self = [NSPanel initWithContentRect:rect 
                          styleMask:NSBorderlessWindowMask
                            backing:buf
                              defer:flag]

…当你想要的是:

self = [super initWithContentRect:rect 
                        styleMask:NSBorderlessWindowMask
                          backing:buf
                            defer:flag]

后者的 PyObjC 等价物是:

self = super().initWithContentRect_styleMask_backing_defer_(rect, NSBorderlessWindowMask, buf, flag)

或者,在 2.x 中:

self = super(HUDWindow, self).initWithContentRect_styleMask_backing_defer_(rect, NSBorderlessWindowMask, buf, flag)

在 ObjC 中,super不是超类,而是一个特殊的魔法对象,它实际上是“作为超类的实例的自我”。它几乎与 Python 3super()或不那么神奇的 Python 2完全一样super,并且 PyObjC 做了额外的工作,这意味着您可以“几乎”忽略它。

你正在做的是(试图)初始化NSPanel类对象(或者可能是 PyObjC 桥的类对象NSPanel,我不确定是哪个),或者甚至可能试图rect初始化NSPanel. 无论您在做什么,它都不可能做任何事情来初始化您的self实例。(并且将结果分配给self在 Python 中与在 ObjC 中做的事情不同。它在 PyObjC 中通常仍然是惯用的风格,但不要被它误导;它的唯一效果是如果你self在初始化方法中再次使用. 无论如何,这充其量只能掩盖 ObjC 中的问题,而不是修复它……)所以,在从这个方法返回之后,你self仍然NSPanel是你调用任何东西之前的已分配但未初始化的实例。

这主要在本教程的早期两阶段实例化中进行了解释。

于 2013-02-26T23:06:55.413 回答