1

大家好

我有一个用 obj-c 和 cocoa 制作的屏幕保护程序。在 OsX 10.6.2 下一切正常,除了以下内容。在我的屏幕保护程序中,我有一个正在运行一些应用程序的 WebView。当我尝试通过 javascript 调用我的 Objective-c 应用程序(屏幕保护程序)时,我收到一个错误并且屏幕保护程序和系统首选项面板崩溃。

系统偏好设置[86666] *** 由于未捕获的异常“NSInvalidArgumentException”而终止应用程序

原因:'-[NSCFArray drain]:无法识别的选择器发送到实例 0x20049b1e0'

在第一掷:( ***调用堆栈
0的CoreFoundation 0x00007fff8123a444 __exceptionPreprocess + 180
1 libobjc.A.dylib 0x00007fff81f130f3 objc_exception_throw + 45
+ 0:2的CoreFoundation 0x00007fff812931c0 + [NSObject的(NSObject的)doesNotRecognizeSelector]
3的CoreFoundation 0x00007fff8120d08f转发+ 751
4的CoreFoundation 0x00007fff812091d8 _CF_forwarding_prep_0 + 232 5 WebCore 0x00007fff847adee0 _ZN3JSC8Bindings12ObjcInstance10virtualEndEv + 48
6 WebCore 0x00007fff8470d71d _ZN3JSC16RuntimeObjectImp18getOwnPropertySlotEPNS_9ExecStateERKNS_10IdentifierERNS_12PropertySlotE + 397
7 JavaScriptCore 0x00007fff80862b66 NK3JSC7JSValue3getEPNS_9ExecStateERKNS_10IdentifierSer4_1
)Proty_10IdentifierS_4_1) ProtySlotE + 397

我知道这看起来像是一些内存泄漏,但正如您将在代码中看到的那样,我实际上几乎没有分配任何对象。

这只发生在我使用屏幕保护程序系统首选项中的“测试”按钮启动屏幕保护程序时。当我通过终端启动屏幕保护程序或者如果它自动启动时,相同的操作(从 javascript 调用 obj-c)可以正常工作。

也许有人有任何想法,错误可能来自哪里。这是实现中的一些代码:

@implementation ScreensaverView

- (id)initWithFrame:(NSRect)frame isPreview:(BOOL)isPreview {

    self = [super initWithFrame:frame isPreview:isPreview];

    if (self) {

        [self setAnimationTimeInterval:-1];
        [self setAutoresizesSubviews:YES];

        // ::::::::::::::::::::::: Init stuff ::::::::::::::::::    

        // init 
        quitFlag = false;
        previewMode = isPreview;

        // find out the path the screensaver bundle
        pMainBundle = [NSBundle bundleForClass:[self class]];
        pBundlePath = [pMainBundle bundlePath];

        // read Info.plist
        infoDict = [pMainBundle infoDictionary];
    }

    return self;
}

- (void)startAnimation
{   
    [super startAnimation];

    // combine: bundle path + filename for screensaver file 
    NSString *pathToScreensaver = [NSString stringWithString:pBundlePath];
    NSString *valueScreensaverFile;

    if(!previewMode)
    {
        valueScreensaverFile = [infoDict objectForKey:@"ScreensaverFile"];
    }
    else 
    {
        valueScreensaverFile = [infoDict objectForKey:@"PreviewFile"];
    }

    // add filename to bundle path
    pathToScreensaver = [pathToScreensaver stringByAppendingString:valueScreensaverFile];

    // complete NSURL to the screensaver file
    NSURL *screensaverUrl = [NSURL fileURLWithPath: pathToScreensaver];

    webView = [WebView alloc];
    [webView initWithFrame:[self frame]];
    [webView setDrawsBackground:NO];

    // delegation policy for interactive mode
    [webView setPolicyDelegate: self];
    [webView setUIDelegate:self];

    // load screensaver
    [[webView mainFrame] loadRequest:[NSURLRequest requestWithURL:screensaverUrl]];

    scriptObject = [webView windowScriptObject];
    [scriptObject setValue:self forKey:@"screensaver"];

    [self addSubview:webView];
}

- (void)stopAnimation
{   
    [[webView mainFrame] stopLoading];
    [webView removeFromSuperview];
    [webView release];
    [super stopAnimation];
}

+ (BOOL)isSelectorExcludedFromWebScript:(SEL)selector 
{       
    if (selector == @selector(quitScreenSaver)) {
        return NO;
    }

    if(selector == @selector(gotoUrl:) ){
        return NO;
    }

    return YES;
}

+(NSString *)webScriptNameForSelector:(SEL)selector
{   
    if(selector == @selector(quitScreenSaver))
    {
        return @"quitNoOpen";
    }

    if(selector == @selector(gotoUrl:))
    {
        return @"openAndQuit";
    }

    return nil;
}

- (void) quitScreenSaver
{
    quitFlag = true;
    [super stopAnimation];
}

- (void) gotoUrl:(NSString *) destinationURL 
{   
    if(destinationURL == NULL)
    {
        return;
    }

    NSString * path    = destinationURL;
    NSURL    * fileURL = [NSURL URLWithString:path];
    [[ NSWorkspace sharedWorkspace ] openURL:fileURL];
    [self quitScreenSaver];
}

@end

我希望这些代码足以让您看到一些问题/解决方案。我真的很感激任何答案。

4

4 回答 4

4

不知何故,正在向 NSCFArray (NSMutableArray) 发送一条“drain”消息,该消息用于 NSAutoreleasePool。

通过实现 NSMutableArray 的 drain 方法,您可能可以获得有关数组是什么的更多信息,因此您可以捕获现在识别的选择器并打印出数组对象的内容。尝试在代码中的某处添加:

@interface NSMutableArray (drain)

- (void) drain;

@end

@implementation NSMutableArray (drain)

- (void) drain
{
   NSLog(@"drain message received by object: %@", self);
}

@end

如果您在控制台中没有看到任何消息,请尝试将上述代码中的“NSMutableArray”更改为“NSObject”。

于 2010-02-12T09:36:52.903 回答
1

需要注意的一件事是,当您通过 System Prefs 中的“测试”按钮启动屏幕保护程序时,您实际上有 2 个屏幕保护程序视图实例在不同线程的同一进程的地址空间中运行。一个(isPreview==YES)是SysPrefs窗口中的小预览(即使启动全屏版本也会继续运行),另一个是全屏版本。它们都在 SysPrefs.app 进程中运行。因此,您必须小心检查所有通知/等。查看它们是否来自您期望的视图实例。

快速浏览一下您发布的代码,我没有看到任何明显的问题,但它可能在其他地方。你在任何地方都使用通知吗?

我在 github 上的 http://github.com/kelan/WikiWalker上放了一个类似的 webview-in-a-screensaver 项目,在那里我最初遇到了一些类似的问题(尽管我没有使用任何 javascript 的东西)。这不是完美的代码,但可能会有所帮助。我还做了一些技巧来将通知转发到 . 请参阅 WWScreenSaverView.{h,m} 的“线程通知支持”部分。

于 2010-02-10T00:37:08.223 回答
1

可以尝试的东西:

  • 打开终端窗口并输入以下行以使用NSZombieEnabled运行系统偏好设置:

env NSZombieEnabled=YES "/Applications/System Preferences.app/Contents/MacOS/System Preferences"

  • 执行导致崩溃的步骤。

  • 运行控制台应用程序,将右上角的过滤器设置为“系统偏好设置”,然后查找 NSZombie 消息。

希望这可以帮助!

于 2010-02-12T06:18:23.197 回答
0

只是为了排除故障,您是否尝试过不发布 WebView?

另外,也许在首先发布之前将 WebView 的代表设置为 nil?

于 2010-02-09T22:53:03.490 回答