基于视图的NSTableView编辑

Zag*_*ggo 16 cocoa nstableview

我不确定我做错了什么.由于我找不到任何关于此的任何其他问题(甚至文档),似乎通常对其他人没有问题.

我只是想尝试使用基于NSTableView的视图来支持编辑它的内容.即应用程序显示带有一列和多行的NSTableView,其中包含带有一些内容的NSTextField.我希望能够(双)单击一个单元格并编辑单元格的内容.所以基本上是基于NSTableView的单元格的正常行为,其中tableView:setObjectValue:forTableColumn:row:实现了该方法.

我分析了Apple的TableViewPlayground示例代码中的Complex TableView示例(支持编辑单元格内容),但我找不到启用编辑的设置/代码/开关.

这是一个简单的示例项目(Xcode 6.1.1,SDK 10.10,基于故事板):

标题:

#import <Cocoa/Cocoa.h>

@interface ViewController : NSViewController

@property (weak) IBOutlet NSTableView *tableView;

@end
Run Code Online (Sandbox Code Playgroud)

执行:

#import "ViewController.h"

@implementation ViewController
{
    NSMutableArray* _content;
}

- (void)viewDidLoad {
    [super viewDidLoad];
    _content = [NSMutableArray array];

    for(NSInteger i = 0; i<10; i++) {
        [_content addObject:[NSString stringWithFormat:@"Item %ld", i]];
    }
}

#pragma mark - NSTableViewDataSource
- (NSInteger)numberOfRowsInTableView:(NSTableView *)aTableView
{
    return _content.count;
}

#pragma mark - NSTableViewDelegate
- (NSView *)tableView:(NSTableView *)tableView viewForTableColumn:(NSTableColumn *)tableColumn row:(NSInteger)row
{
    NSTableCellView* cell = [tableView makeViewWithIdentifier:@"CellView" owner:self];
    cell.textField.stringValue = _content[row];
    return cell;
}

- (IBAction)endEditingText:(id)sender {
    NSInteger row = [_tableView rowForView:sender];
    if (row != -1) {
        _content[row] = [sender stringValue];
    }
}
@end
Run Code Online (Sandbox Code Playgroud)

故事板文件如下所示: 故事板

表视图的数据源和委托设置为视图控制器.运行此应用程序时,表视图显示10个测试行,但无法编辑其中一行.

这是为什么?我在这里想念的是什么?

我仔细检查了NSTableView的所有属性(及其内容)与Apple的TableViewPlayground示例中的相同.经过几个小时的搜索文档和互联网有用的提示没有任何成功,我有点沮丧.您在基于视图的NSTableViews上找到的所有内容都是不可编辑的样本,或者是关于可编辑内容的非常模糊的信息.当然,在可编辑的基于单元格的NSTableViews上有大量的信息,文档和示例......

我的示例项目的zip可以在这里下载: TableTest.zip

7st*_*tud 48

即使编辑基于NSTableView 的视图的所有部分都存在于问题和答案中,我仍然无法将它们整合在一起.以下演示使用Xcode 6.3.2在Swift中进行,但对于Objective-C cavemen/womens应该很容易理解.最后是一个完整的代码清单.

我们从这里开始:

NSTableViewDataSource协议参考

设置值

- tableView:setObjectValue:forTableColumn:row:

Swift:  
optional func tableView(_ aTableView: NSTableView,
              setObjectValue anObject: AnyObject?,
              forTableColumn aTableColumn: NSTableColumn?,
              row rowIndex: Int)

Objective-C:
- (void)tableView:(NSTableView *)aTableView
   setObjectValue:(id)anObject
   forTableColumn:(NSTableColumn *)aTableColumn
              row:(NSInteger)rowIndex
Run Code Online (Sandbox Code Playgroud)

讨论:此方法适用于基于单元格的表视图,不能与基于视图的表视图一起使用.在基于视图的表中,使用target/action设置视图单元格中的每个项目.

如果你像我一样,你搜索了NSTableViewDelegate和NSTableViewDataSource协议,寻找某种编辑方法来使用.但是,上面引用的讨论告诉你事情要简单得多.

1)如果你在Xcode中查看TableView 的文档大纲,你会看到如下内容:

在此输入图像描述

TableView中的单元格由a表示Table Cell View.单元格包含一些项目,默认情况下,其中一个项目是NSTextField.但是文档大纲中的NSTextField在哪里?!好吧,文档大纲中的控件有一个图标,看起来就像名字旁边的滑块.看一看.在单元格内部,您会看到旁边有滑块图标的东西.并且,如果您在文档大纲中选择该行,然后打开Identity Inspector,您将看到它是NSTextField:

在此输入图像描述

您可以认为这只是一个普通的旧NSTextField.

实现NSTableViewDataSource协议方法时:

import Cocoa

class MainWindowController: NSWindowController,
                            NSTableViewDataSource {
    ...
    ...
    var items: [String] = []  //The data source: an array of String's
    ...
    ...

    // MARK: NSTableViewDataSource protocol methods

    func numberOfRowsInTableView(tableView: NSTableView) -> Int {
        return items.count
    }

    func tableView(tableView: NSTableView,
                   objectValueForTableColumn tableColumn: NSTableColumn?,
                   row: Int) -> AnyObject? {
        return items[row]
    }

}
Run Code Online (Sandbox Code Playgroud)

... TableView获取第二个方法返回的值,并将其分配给objectValue外部命名的属性Table Cell View- 换句话说,TableView不使用返回的值来设置NSTextField(即内部Table View Cell).这意味着您的数据源项目不会显示在TableView中,因为NSTextField是显示项目的内容.为了设置的NSTextField的值,你需要连接,或绑定的的NSTextField的value到objectValue财产.您可以在Bindings Inspector中执行此操作:

警告:Bind to在选择要绑定的对象之前,请确保不选中该复选框.如果先选中该复选框,则会在文档大纲中插入一个对象,如果您没有注意到,则在运行程序时会出现错误.如果您不小心Bind to首先选中了复选框,请确保删除文档大纲中自动添加的对象.Xcode为添加的对象创建一个单独的部分,因此很容易在文档大纲中找到它.

在此输入图像描述

2)回溯片刻,您可能熟悉将按钮连接到动作方法,此后如果单击按钮,动作方法将执行.另一方面,对于NSTextField,您通常会声明一个IBOutlet,然后使用它来获取或设置NSTextField stringValue.

但是,NSTextField也可以触发执行操作方法.什??!但你不能像按钮一样点击NSTextfield!然而,NSTextField有一个触发器,就像一个按钮单击,这将导致执行一个动作方法,触发器是:完成编辑NSTextField.NSTextField如何知道您何时完成编辑?有两种方法:

  1. 你点击Return.

  2. 你点击其他一些控件.

您可以在"属性"检查器中选择触发器:

在此输入图像描述

3)正如@Paul Patterson在他的回答中所示,接下来您需要做的是在属性检查器中将NSTextField设置BehaviorEditable.

4)然后将NSTextField连接到要执行的操作方法.如果您还没有使用以下技巧将控件连接到操作方法,那么您应该尝试一下:

在Project Navigator中选择.xib文件,以便显示窗口及其控件.然后单击Assistant Editor(最右侧Xcode窗口顶部的two_interlocking_rings图标) - 将显示您的Controller文件(如果显示了其他文件,则使用跳转栏导航到您的Controller文件) .然后按住Control键并从NSTextField(在文档大纲中)拖动到您要在其中创建操作方法的Controller文件中的位置:

在此输入图像描述

当您发布时,您将看到此弹出窗口:

在此输入图像描述

如果输入的信息与显示的信息相同,则会在文件中输入以下代码:

@IBAction func onEnterInTextField(sender: NSTextField) {

}
Run Code Online (Sandbox Code Playgroud)

并且......已经完成了NSTextField和action方法之间的连接.(您也可以使用这些步骤创建和连接IBOutlet.)

5)在action方法中,您可以从TableView获取当前选定的行,即刚刚编辑过的行:

@IBAction func onEnterInTextField(sender: NSTextField) {    
    let selectedRowNumber = tableView.selectedRow  //tableView is an IBOutlet connected to the NSTableView

}
Run Code Online (Sandbox Code Playgroud)

然后我对如何在TableView中获取所选行的文本感到困惑,然后回到我通过TableView和协议方法查看的文档.但是,我们都知道如何获取NSTextField的stringValue,对吧?该发件人是你编辑的NSTextField:

@IBAction func onEnterInTextField(sender: NSTextField) {
    let selectedRowNumber = tableView.selectedRow  //My Controller has  an IBOutlet property named tableView which is connected to the TableView 

    if selectedRowNumber != -1 {  //-1 is returned when no row is selected in the TableView
        items[selectedRowNumber] = sender.stringValue  //items is the data source, which is an array of Strings to be displayed in the TableView
    }

}
Run Code Online (Sandbox Code Playgroud)

如果未在数据源中插入新值,则下次TableView需要显示行时,将再次显示原始值,覆盖已编辑的更改.请记住,TableView从数据源检索值 - 而不是NSTextField.然后NSTextField显示TableView分配给单元格的objectValue属性的任何值.

最后一件事:我收到一条警告说我无法将NSTextField连接到类中的动作,如果该类不是TableView的委托....所以我将TableView的委托出口连接到File的所有者:

在此输入图像描述

我之前已经将File的Owner设置为我的Controller(= MainWindowController),所以在我建立连接后,MainWindowController(包含NSTextField的action方法)成为TableView的委托,警告消失了.

随机提示:

1)我发现开始编辑NSTextField 最简单的方法是在TableView中选择一行,然后点击Return.

2)默认情况下,NSTableView有两列.如果选择文档大纲中的一列,然后单击键盘上的Delete,则可以创建一个列表 - 但TableView仍然显示列分隔符,因此看起来仍然有两列.要删除列分隔符,请Bordered Scroll View - Table View在文档大纲中选择,然后拖动其中一个角来调整TableView的大小 - 单个列将立即调整大小以占用所有可用空间.

感谢步骤#1和#2,以及随机提示#2:OS X的可可编程(2015年第5版).

完整代码清单:

//
//  MainWindowController.swift
//  ToDo
//

//import Foundation
import Cocoa

class MainWindowController: NSWindowController,
                            NSTableViewDataSource {

    //@IBOutlet var window: NSWindow? -- inherited from NSWindowController
    @IBOutlet weak var textField: NSTextField!
    @IBOutlet weak var tableView: NSTableView!

    var items: [String] = []  //The data source: an array of String's

    override var windowNibName: String {
        return "MainWindow"
    }

    @IBAction func onclickAddButton(sender: NSButton) {
        items.append(textField.stringValue)
        tableView.reloadData()  //Displays the new item in the TableView

    }

    @IBAction func onEnterInTextField(sender: NSTextField) {
        let selectedRowNumber = tableView.selectedRow

        if selectedRowNumber != -1 {
            items[selectedRowNumber] = sender.stringValue
        }
    }


    // MARK: NSTableViewDataSource protocol methods

    func numberOfRowsInTableView(tableView: NSTableView) -> Int {
        return items.count
    }

    func tableView(tableView: NSTableView,
                   objectValueForTableColumn tableColumn: NSTableColumn?,
                   row: Int) -> AnyObject? {
        return items[row]
    }

 }
Run Code Online (Sandbox Code Playgroud)

Connections Inspector显示File的Owner(= MainWindowController)的所有连接:

在此输入图像描述

  • 有时,通常在表首次出现之后,编辑的单元格可能与`tableView.selectedRow`返回的行不在同一行.我发现`tableView.row(for:sender)`更可靠. (4认同)

Pau*_*son 27

默认情况下,每个单元格(实例NSTableCellView)都有一个NSTextField坐在上面.当您编辑单元格时,您实际编辑的是此文本字段.Interface Builder使此文本字段不可编辑:

在此输入图像描述

您需要做的就是将" 行为"弹出窗口设置为Editable.现在你可以打回或编辑文本字段 -click.

  • 如何双击进行编辑? (3认同)