Подтвердить что ты не робот

Ответ на события мыши в текстовом поле в представлении таблицы на основе представления

У меня есть текстовые поля внутри пользовательского представления внутри NSOutlineView. Для редактирования одной из этих ячеек требуется один клик, пауза и еще один клик. Первый одиночный клик выбирает строку вида таблицы, а второй одиночный клик рисует курсор в поле. Двойной щелчок по ячейке, который позволяет редактировать в виде таблицы на основе таблицы, выбирает только строку.

Поведение, которое я хочу: один клик, чтобы изменить выбор и изменить.

Что мне нужно переопределить, чтобы получить это поведение?

Я прочитал несколько других сообщений:

  • NSTextField flyweight pattern, похоже, не будет применяться к представлениям таблиц на основе представления, где все представления ячеек создаются из nibs.
  • Я попробовал подклассификацию NSTextField как это решение описывает, но мой переопределенный метод mouseDown не вызывается. Вызывается переопределенные awakeFromNib и viewWillDraw (упомянутые в этом сообщении). Конечно, mouseDown вызывается, если я помещаю текстовое поле где-то вне табличного представления.

Для сравнения, a NSSegmentedControl в моем представлении ячейки меняет свое значение без предварительного выбора строки.


Здесь рабочее решение адаптировано из принятого ответа:

В подклассе вида:

-(void)mouseDown:(NSEvent *)theEvent {
    [super mouseDown:theEvent];

    // Forward the click to the row cell view
    NSPoint selfPoint = [self convertPoint:theEvent.locationInWindow fromView:nil];
    NSInteger row = [self rowAtPoint:selfPoint];
    if (row>=0) [(CellViewSubclass *)[self viewAtColumn:0 row:row makeIfNecessary:NO]
            mouseDownForTextFields:theEvent];
}

В подклассе ячеек таблицы:

// Respond to clicks within text fields only, because other clicks will be duplicates of events passed to mouseDown
- (void)mouseDownForTextFields:(NSEvent *)theEvent {
    // If shift or command are being held, we're selecting rows, so ignore
    if ((NSCommandKeyMask | NSShiftKeyMask) & [theEvent modifierFlags]) return;
    NSPoint selfPoint = [self convertPoint:theEvent.locationInWindow fromView:nil];
    for (NSView *subview in [self subviews])
        if ([subview isKindOfClass:[NSTextField class]])
            if (NSPointInRect(selfPoint, [subview frame]))
                [[self window] makeFirstResponder:subview];
}
4b9b3361

Ответ 1

Я попытаюсь вернуть услугу... Подкласс NSOutlineView и переопределить -mouseDown: следующим образом:

- (void)mouseDown:(NSEvent *)theEvent {
    [super mouseDown:theEvent];

    // Only take effect for double clicks; remove to allow for single clicks
    if (theEvent.clickCount < 2) {
        return;
    }

    // Get the row on which the user clicked
    NSPoint localPoint = [self convertPoint:theEvent.locationInWindow
                                   fromView:nil];
    NSInteger row = [self rowAtPoint:localPoint];

    // If the user didn't click on a row, we're done
    if (row < 0) {
        return;
    }

    // Get the view clicked on
    NSTableCellView *view = [self viewAtColumn:0 row:row makeIfNecessary:NO];

    // If the field can be edited, pop the editor into edit mode
    if (view.textField.isEditable) {
        [[view window] makeFirstResponder:view.textField];
    }
}

Ответ 2

Была та же проблема. После большой борьбы он волшебным образом работал, когда я выбрал None по умолчанию Regular (другая опция Source List) для опции Highlight в представлении таблицы в IB!

Другим вариантом является решение на fooobar.com/questions/199332/..., которое выглядит более конкретным, но немного взломанным по сравнению с этим.

Ответ 3

Я написал следующее, чтобы поддержать случай, когда у вас есть более сложный NSTableViewCell с несколькими текстовыми полями или где текстовое поле не занимает всю ячейку. Здесь есть трюк, когда вы переключаете значения y, потому что, когда вы переключаетесь между NSOutlineView или NSTableView и NSTableCellViews, система координат перевернута.

- (void)mouseDown:(NSEvent *)theEvent
{
    [super mouseDown: theEvent];

    NSPoint thePoint = [self.window.contentView convertPoint: theEvent.locationInWindow
                                                      toView: self];
    NSInteger row = [self rowAtPoint: thePoint];
    if (row != -1) {
        NSView *view = [self viewAtColumn: 0
                                      row: row
                          makeIfNecessary: NO];

        thePoint = [view convertPoint: thePoint
                             fromView: self];
        if ([view isFlipped] != [self isFlipped])
            thePoint.y = RectGetHeight(view.bounds) - thePoint.y;

        view = [view hitTest: thePoint];
        if ([view isKindOfClass: [NSTextField class]]) {
            NSTextField *textField = (NSTextField *)view;
            if (textField.isEnabled && textField.window.firstResponder != textField)

                dispatch_async(dispatch_get_main_queue(), ^{
                    [textField selectText: nil];
                });
        }
    }
}

Ответ 4

Вы действительно хотите переопределить validateProposedFirstResponder и разрешить конкретный первый ответчик (или нет) в зависимости от вашей логики. Реализация в NSTableView (вроде) вроде этого (я переписываю ее как псевдокод):

- (BOOL)validateProposedFirstResponder:(NSResponder *)responder forEvent:(NSEvent *)event {

// We want to not do anything for the following conditions:
// 1. We aren't view based (sometimes people have subviews in tables when they aren't view based)
// 2. The responder to valididate is ourselves (we send this up the chain, in case we are in another tableview)
// 3. We don't have a selection highlight style; in that case, we just let things go through, since the user can't appear to select anything anyways.
if (!isViewBased || responder == self || [self selectionHighlightStyle] == NSTableViewSelectionHighlightStyleNone) {
    return [super validateProposedFirstResponder:responder forEvent:event];
}

if (![responder isKindOfClass:[NSControl class]]) {
    // Let any non-control become first responder whenever it wants
    result = YES;
    // Exclude NSTableCellView. 
    if ([responder isKindOfClass:[NSTableCellView class]]) {
        result = NO;
    }

} else if ([responder isKindOfClass:[NSButton class]]) {
    // Let all buttons go through; this would be caught later on in our hit testing, but we also do it here to make it cleaner and easier to read what we want. We want buttons to track at anytime without any restrictions. They are always valid to become the first responder. Text editing isn't.
    result = YES;
} else if (event == nil) {
    // If we don't have any event, then we will consider it valid only if it is already the first responder
    NSResponder *currentResponder = self.window.firstResponder;
    if (currentResponder != nil && [currentResponder isKindOfClass:[NSView class]] && [(NSView *)currentResponder isDescendantOf:(NSView *)responder]) {
        result = YES;
    }
} else {
    if ([event type] == NSEventTypeLeftMouseDown || [event type] == NSEventTypeRightMouseDown) {
        // If it was a double click, and we have a double action, then send that to the table
        if ([self doubleAction] != NULL && [event clickCount] > 1) {
           [cancel the first responder delay];
        }
   ...
          The code here checks to see if the text field 
        cell had text hit. If it did, it attempts to edit it on a delay. 
        Editing is simply making that NSTextField the first responder.
        ...

     }

Ответ 5

Просто хочу указать, что если все, что вы хотите, редактирует только (т.е. в таблице без выбора), переопределение -hitTest: кажется более простым и более Cocoa -like:

- (NSView *)hitTest:(NSPoint)aPoint
{
    NSInteger column = [self columnAtPoint: aPoint];
    NSInteger row = [self rowAtPoint: aPoint];

    // Give cell view a chance to override table hit testing
    if (row != -1 && column != -1) {
        NSView *cell = [self viewAtColumn:column row:row makeIfNecessary:NO];

        // Use cell frame, since convertPoint: doesn't always seem to work.
        NSRect frame = [self frameOfCellAtColumn:column row:row];
        NSView *hit = [cell hitTest: NSMakePoint(aPoint.x + frame.origin.x, aPoint.y + frame.origin.y)];

        if (hit)
            return hit;
    }

    // Default implementation
    return [super hitTest: aPoint];
}