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

时间:2013-09-01 16:35:38

标签: cocoa swift cell nstableview mouseclick-event

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

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

  1. 它仅在选择发生变化时触发,而我希望每次鼠标点击,即使它在当前选定的行上。
  2. 调用我的委托时,NSTableView的clickedRow和clickedColumn属性都是-1。
  3. 有没有更好(和更正确)的方法呢?

6 个答案:

答案 0 :(得分:39)

有一种简单的方法。

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

tableView.action = #selector(onItemClicked)

然后

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

答案 1 :(得分:25)

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

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

MyTableView.h

@protocol ExtendedTableViewDelegate <NSObject>

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

@end

@interface MyTableView : NSTableView

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

@end

MyTableView.m

处理鼠标按下事件(注意,当用户点击外部时,不会调用委托回调,也许您也想处理它,在这种情况下,只需注释掉条件“{{ 1}}“)

if (clickedRow != -1)

让您的WC,VC符合ExtendedTableViewDelegate

- (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];
    }
}

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

MyTableView.m

中的某个地方
@interface MyViewController : DocumentBaseViewController<ExtendedTableViewDelegate, NSTableViewDelegate,  NSTableViewDataSource>

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

self.myTableView.extendedDelegate = self

答案 2 :(得分:8)

我更愿意这样做。

覆盖

-(BOOL)tableView:(NSTableView *)tableView shouldSelectRow:(NSInteger)row;

提供超级实施;

RequiredRow = row;
RequiredColumn = [tableView clickedColumn];

希望这有帮助。

答案 3 :(得分: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)
}

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

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

@nonobjc属性使关于它接近didClick的警告无声。

答案 4 :(得分:3)

以防万一有人在 SWIFT 和/或 NSOutlineView 中寻找它。

基于@Peter Lapisu说明。

class MYOutlineViewDelegate: NSOutlineView, NSOutlineViewDelegate,NSOutlineViewDataSource{
//....
}    
extension MYOutlineViewDelegate{
    func outlineView(outlineView: NSOutlineView, didClickTableRow item: AnyObject?) {
        //Click stuff
    }

    override func mouseDown(theEvent: NSEvent) {
        let globalLocation:NSPoint  = theEvent.locationInWindow
        let localLocation:NSPoint  = self.convertPoint(globalLocation, fromView: nil)
        let clickedRow:Int = self.rowAtPoint(localLocation)

        super.mouseDown(theEvent)

        if (clickedRow != -1) {
            self.outlineView(self, didClickTableRow: self.itemAtRow(clickedRow))
        }
    }}

答案 5 :(得分:-1)

查看 tableViewSelectionIsChanging 通知,以下是NSTableView.h的注释

/ * 可选 - 在即将更改选择时调用,但请注意,tableViewSelectionIsChanging:仅在鼠标事件更改选择而不是键盘事件时调用。  * /

我承认这可能不是关联鼠标点击的最可靠方法,但这是另一个需要调查的方面,因为您对鼠标点击感兴趣。