我正在尝试检测NSTableView中何时发生鼠标单击,以及何时单击,以确定单击的单元格的行和列。
到目前为止,我已尝试使用NSTableViewSelectionDidChangeNotification,但有两个问题:
有没有更好(和更正确)的方法呢?
答案 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:仅在鼠标事件更改选择而不是键盘事件时调用。 * /
我承认这可能不是关联鼠标点击的最可靠方法,但这是另一个需要调查的方面,因为您对鼠标点击感兴趣。