NSTableView:检测鼠标单击以及行和列

bri*_*ght 20 cocoa cell nstableview mouseclick-event swift

我正在尝试检测何时在NSTableView中发生鼠标单击,以及何时单击,以确定单击的单元格的行和列.

到目前为止,我已尝试使用NSTableViewSelectionDidChangeNotification,但有两个问题:

  1. 它仅在选择更改时触发,而我希望每次鼠标单击,即使它在当前选定的行上.
  2. 调用我的委托时,NSTableView的clickedRow和clickedColumn属性都是-1.

这样做有更好(和更正确)的方法吗?

lon*_*kai 45

有一个简单的方法.

在macOS 10.12.2和Xcode 8.2.1上使用Swift 3.0.2进行测试

tableView.action = #selector(onItemClicked)
Run Code Online (Sandbox Code Playgroud)

然后

@objc private func onItemClicked() {
    print("row \(tableView.clickedRow), col \(tableView.clickedColumn) clicked")
}
Run Code Online (Sandbox Code Playgroud)

  • 只要确保也设置了“ tableView.target”即可。 (3认同)
  • 似乎是最佳答案。 (2认同)
  • 杰出的!这确实是最好的答案 (2认同)

Pet*_*isu 26

捕获用户单击一行(仅当用户单击某行时,而不是以编程方式选择它时):

对您的NSTableView进行子类化并声明协议

MyTableView.h

@protocol ExtendedTableViewDelegate <NSObject>

- (void)tableView:(NSTableView *)tableView didClickedRow:(NSInteger)row;

@end

@interface MyTableView : NSTableView

@property (nonatomic, weak) id<ExtendedTableViewDelegate> extendedDelegate;

@end
Run Code Online (Sandbox Code Playgroud)

MyTableView.m

处理鼠标按下事件(注意,当用户点击外部时不会调用委托回调,也许你也想处理它,在这种情况下,只需注释条件" if (clickedRow != -1)")

- (void)mouseDown:(NSEvent *)theEvent {

    NSPoint globalLocation = [theEvent locationInWindow];
    NSPoint localLocation = [self convertPoint:globalLocation fromView:nil];
    NSInteger clickedRow = [self rowAtPoint:localLocation];

    [super mouseDown:theEvent];

    if (clickedRow != -1) {
        [self.extendedDelegate tableView:self didClickedRow:clickedRow];
    }
}
Run Code Online (Sandbox Code Playgroud)

使您的WC,VC符合ExtendedTableViewDelegate.

@interface MyViewController : DocumentBaseViewController<ExtendedTableViewDelegate, NSTableViewDelegate,  NSTableViewDataSource>
Run Code Online (Sandbox Code Playgroud)

将MyTableView的extendedDelegate设置为WC,VC(MyViewController)

MyTableView.m中的某个地方

self.myTableView.extendedDelegate = self
Run Code Online (Sandbox Code Playgroud)

在委托中实现回调(MyViewController.m)

- (void)tableView:(NSTableView *)tableView didClickedRow:(NSInteger)row {
    // have fun
}
Run Code Online (Sandbox Code Playgroud)


Suh*_*hal 8

我更喜欢这样做.

覆盖

-(BOOL)tableView:(NSTableView *)tableView shouldSelectRow:(NSInteger)row;
Run Code Online (Sandbox Code Playgroud)

提供超级实施;

RequiredRow = row;
RequiredColumn = [tableView clickedColumn];
Run Code Online (Sandbox Code Playgroud)

希望这可以帮助.

  • 这不仅是由用户操作触发的,如问题中所述"我正在尝试检测NSTableView中何时发生鼠标单击,以及何时发生,以确定单击的单元格的行和列"但每次选择都会以编程方式改变 (2认同)

小智 7

如果有人正在寻找Peter Lapisu的答案的Swift 3版本。

添加NSTableView的扩展名(NSTableView + Clickable.swift):

import Foundation
import Cocoa

extension NSTableView {
    open override func mouseDown(with event: NSEvent) {
        let globalLocation = event.locationInWindow
        let localLocation = self.convert(globalLocation, to: nil)
        let clickedRow = self.row(at: localLocation)

        super.mouseDown(with: event)

        if (clickedRow != -1) {
            (self.delegate as? NSTableViewClickableDelegate)?.tableView(self, didClickRow: clickedRow)
        }
    }
}

protocol NSTableViewClickableDelegate: NSTableViewDelegate {
    func tableView(_ tableView: NSTableView, didClickRow row: Int)
}
Run Code Online (Sandbox Code Playgroud)

然后,要使用它,请确保实现新的委托协议:

extension MyViewController: NSTableViewClickableDelegate {
    @nonobjc func tableView(_ tableView: NSTableView, didClickRow row: Int) {
        Swift.print("Clicked row \(row)")
    }
}
Run Code Online (Sandbox Code Playgroud)

@nonobjc属性使有关它接近didClick的警告静音。