4

我一直在与这个问题作斗争很长一段时间。我正在开发一个文件复制管理器模块,到目前为止,除了取消按钮之外,我已经能够使一切正常工作。出于某种原因,当我单击特定行的取消按钮时,按钮操作同时针对多行。

文件复制管理器

经过几天的研究,我能够使对象成功取消该行所代表的操作,使用:

-(IBAction)btnCancelOperationClick:(id)sender {
    NSInteger row = [_tableView rowForView:sender];
    if (row != -1) {
        FileCopyOperation *opr = [_fileCopyOperations objectAtIndex:row];
        [opr cancel];
        [_fileCopyOperations removeObjectAtIndex:row];
        [_tableView removeRowsAtIndexes:[NSIndexSet indexSetWithIndex:row] withAnimation:NSTableViewAnimationEffectFade];
    }
}

这很好用,我可以取消操作并相应地更新我的表。其他一切都按预期工作,但我的代码或绑定肯定有问题。我正在从笔尖加载此单元格,然后使用以下命令注册此笔尖:

[_tableView registerNib:[[NSNib alloc]initWithNibNamed:@"FileCopyCell" bundle:nil] forIdentifier:@"FileCopyCell"];

我将 QueueController 设为文件的所有者,并像这样连接所有内容:

绑定

如果有人能指出我正确的方向以使这件事正常工作,我将不胜感激。提前致谢!

编辑以添加更多代码示例。

-(NSView *)tableView:(NSTableView *)tableView viewForTableColumn:(NSTableColumn *)tableColumn row:(NSInteger)row {
    FileCopyCell *cell = [tableView makeViewWithIdentifier:@"FileCopyCell" owner:self];
    FileCopyOperation *opr = [_fileCopyOperations objectAtIndex:row];

    [cell.fileName setStringValue:[NSString stringWithFormat:@"Copying \"%@\"",opr.fName]];
    [cell.progressBar setDoubleValue:((opr.bWritten.doubleValue / opr.fSize.doubleValue) * 100)];
    [cell.totalBytes setStringValue:[NSString stringWithFormat:@"of %@",[NSByteCountFormatter stringFromByteCount:opr.fSize.longLongValue countStyle:NSByteCountFormatterCountStyleFile]]];
    [cell.status setStringValue:[NSString stringWithFormat:@"%@",[NSByteCountFormatter stringFromByteCount:opr.bWritten.longLongValue countStyle:NSByteCountFormatterCountStyleFile]]];
    [cell.icon setImage:[[NSWorkspace sharedWorkspace]iconForFile:opr.srcURL.path]];
    [cell.cancelButton setTarget:self];
    return cell;
}

-(void)windowDidLoad {
    [super windowDidLoad];
    _fileCopyOperations = [NSMutableArray new];
    windowFrame = [self.window frame];
    rows = 0;

    [_tableView registerNib:[[NSNib alloc]initWithNibNamed:@"FileCopyCell" bundle:nil] forIdentifier:@"FileCopyCell"];

    if (!fileCopyManager) {
        fileCopyManager = [FileCopyManager sharedFileCopyManager];
        [fileCopyManager.fileCopyQueue addObserver:self forKeyPath:@"operationCount" options:NSKeyValueObservingOptionNew context:(void*)fileCopyManager];
    }

    [_scrollView setHasHorizontalScroller:NO];
    [_scrollView setHasVerticalScroller:NO];
}
4

2 回答 2

4

最好的方法是子类化NSTableCellView并让它处理自己的动作和表示的对象。例如,代表Foo实例的单元格可能有两个属性:foofooController。当调用 ( nonatomic)foo设置器时,单元格可以更新自己的 UI 以表示传递的Foo. 当Foo控制器创建表格单元格时,它可以实例化一个FooCell实例,将自己设置为fooController并分配Foo实例并让单元格自己处理。取消按钮的目标可以是它自己的单元格,当-cancel:动作被调用时,它可以告诉它fooController要做什么(因为Foo控制器负责更新队列和表视图),并且由于它具有对其的引用foo,因此它可以通过某种-cancelFoo:(Foo *)theFoo方法将其传递给控制器​​,而无需依赖控制器来查找其索引(如果您可能不准确)重新为出现和消失的行设置动画,或者用户正在快速取消一行,但它们的删除被延迟并异步更新)。

干净整洁。整齐有序的遏制/职责分离。cell 处理自己的 UI 更新和操作,并且知道自己的 foo;foo 控制器处理其 foo 集合、其表视图以及将 foos 分配给 foo 单元格。

于 2015-04-17T13:53:22.557 回答
1

感谢 Joshua Nozzi,按照他的建议,我将按钮操作从控制器移到了单元类。在单元格类中,我使用以下方法访问表示的对象并发送[operation cancel]消息。

-(IBAction)cancelOpr:(id)sender {
    NSButton *button = (NSButton*)sender;
    FileCopyOperation *opr = [(NSTableCellView*)[button superview]objectValue];
    [opr cancel];
    // This code calls the controller [removeObject:] method that takes care of cleaning everything out and updates the GUI accordingly.
    AppDelegate *ad = (AppDelegate*)[[NSApplication sharedApplication]delegate];
    QueueWindowController *qc = [ad getQueueController];
    [qc.fileCopyOperations performSelectorOnMainThread:@selector(removeObject:) withObject:opr waitUntilDone:YES];
}

你可以在这里得到完整的项目。

在此处输入图像描述

于 2015-04-20T12:29:37.257 回答