3

我有NSOpenPanel一个accessoryView; 在此视图中,用户选择几个单选按钮来更改允许的类型。当面板打开时,正确的文件被启用,另一个被禁用。好的,很好。

现在用户更改单选按钮,viewControlleraccessoryView 观察单选按钮矩阵的变化并随之改变allowedTypesNSOpenPanel。

之后,根据 Apple 文档,它调用-validateVisibleColumns,但面板中没有任何可见的变化。也就是说:正确的文件似乎已禁用:我可以选择它们,但它们是灰色的!另一个错误的效果:我选择了一个文件(启用),更改文件类型,(现在错误的)文件保持选中状态,启用了 OK 按钮:但这是错误的文件类型!似乎发生了变化,但界面不知道!

我的代码是(选择绑定到单选按钮的矩阵):

- (void)observeValueForKeyPath..... 
{
    NSString *extension = (self.selected==0) ? @"txt" : @"xml";
    [thePanel setAllowedFileTypes:@[extension, [extension uppercaseString]]];
    [thePanel validateVisibleColumns];
}

我首先尝试插入一个呼叫

[thePanel displayIfNeeded]

然后我尝试了

[thePanel contentView] setNeedsDisplay]

没有结果。我还尝试实现面板委托方法panel:shouldEnableURL:,应该由validateVisibleColumns:我刚刚发现它只被调用了一次,在 NSOpenPanel 的开头。

有人可以知道为什么会这样吗?我用沙盒和非沙盒应用程序尝试了所有这些,没有区别。我正在使用 10.8 sdk 进行 ML 开发。

编辑

到目前为止,避免该问题的唯一方法是实现panel:validateURL:error,但这是在用户单击“打开”后调用的,这非常糟糕。

4

3 回答 3

5

我有完全相同的问题,在 10.9 下,非沙盒,并且在这一天的大部分时间里都在试图找到解决方案!

经过大量修补和深入研究构成NSOpenPanelNSSavePanel真的)的各种类后,我确实找到了一种方法来强制基础表刷新自身:

id table = [[[[[[[[[[[[_openPanel contentView] subviews][4] subviews][0] subviews][0] subviews][0] subviews][7] subviews][0] subviews][1] subviews][0] subviews][0] subviews][0] subviews][2];
[table reloadData];

当然,编写此 hack 的最佳方法是遍历子视图列表,确保找到正确的类并最终缓存最终表视图以供后续reloadData调用。

我知道,我知道,这是一个非常难看的问题,但是,除了“提交错误报告”之外,我似乎找不到任何其他答案来解决这个问题。从我可以看到网上人们自 1.8 以来一直在做的事情!:(

编辑: 这是我现在用来使我的 NSOpenPanel 在 10.9 下正常运行的代码:

- (id) openPanelFindTable: (NSArray*)subviews;
{
    id table = nil;

    for (id view in subviews) {
        if ([[view className] isEqualToString: @"FI_TListView"]) {
            table = view;
            break;
        } else {
            table = [self openPanelFindTable: [view subviews]];
            if (table != nil) break;
        }
    }

    return table;
}


- (void) refreshOpenPanel
{
    if (_openPanelTableHack == nil)
        _openPanelTableHack = [self openPanelFindTable: [[_openPanel contentView] subviews]];
    [_openPanelTableHack reloadData];
    [_openPanel validateVisibleColumns];
}

这段代码需要两个实例变量_openPanel并被_openPanelTableHack声明才能工作。我声明_openPanelNSOpenPanel*并且_openPanelTableHack被声明为id

现在,[_openPanel validateVisibleColumns]我没有打电话[self refreshOpenPanel]来强制面板按预期更新文件名。我尝试在创建 NSOpenPanel 时缓存表格视图,但是,似乎一旦您“运行”面板,表格视图就会更改,因此我必须在第一次更新时缓存它。

同样,这是一个 GIANT hack,但是,我不知道 Apple 需要多长时间才能解决附件视图和文件面板的问题,所以现在,这可行。

如果有人有任何其他不是很大的解决方案,请分享!;)

于 2014-01-22T17:33:19.283 回答
1

只需升级到 xcode 6.3(在 Yosemite 10.10.3 上)就可以了。Apple 修复了这个错误(不再需要 Eidola Hack)。

于 2015-04-09T15:35:26.663 回答
1

Eidola 解决方案的快速实现。最大的区别是我搜索的是 NSBrowser(子)类而不是特定的类名。在 10.10 上测试(未沙盒)。

private weak var panelBrowser : NSBrowser? //avoid strong reference cycle
func reloadBrowser()
{
    if let assumedBrowser = panelBrowser
    {
        assumedBrowser.reloadColumn(assumedBrowser.lastColumn)
    }
    else if let searchResult = self.locateBrowser(self.panel?.contentView as! NSView)
    {
        searchResult.reloadColumn(searchResult.lastColumn)
        self.panelBrowser = searchResult //hang on to result
    }
    else
    {
        assertionFailure("browser not found")
    }
}

//recursive search function
private func locateBrowser(view: NSView) -> NSBrowser?
{
    for subview in view.subviews as! [NSView]
    {
        if subview is NSBrowser
        {
            return subview as? NSBrowser
        }
        else if let result = locateBrowser(subview)
        {
            return result
        }
    }
    return nil
}

编辑:

好的,所以上面的代码不会一直工作。如果它不起作用并且选择了一个文件(您可以查看详细信息/预览),那么您必须将最后一列重新加载到一列而不是最后一列。重新加载最后两列(确保至少有 2 列)或重新加载所有列。

第二个问题:如果您重新加载列,那么您将失去选择。是的,所选文件/目录仍将突出显示,但面板不会返回正确的 URL。

现在我正在使用这个功能:

func reloadBrowser()
{
    //obtain browser
    if self.panelBrowser == nil
    {
        self.panelBrowser = self.locateBrowser(self.panel?.contentView as! NSView)
    }
    assert(panelBrowser != nil, "browser not found")

    //reload browser
    let panelSelectionPatch = panelBrowser.selectionIndexPaths //otherwise the panel return the wrong urls
    if panelBrowser.lastColumn > 0
    {
        panelBrowser.reloadColumn(panelBrowser.lastColumn-1)
    }
    panelBrowser.reloadColumn(panelBrowser.lastColumn)
    panelBrowser.selectionIndexPaths = panelSelectionPatch
}
于 2015-03-13T11:16:41.573 回答