Как добавить контекстно-зависимое меню в NSOutlineView (т. Е. По меню правой кнопки мыши) - PullRequest
21 голосов
/ 21 августа 2009

Как добавить возможность щелчка правой кнопкой мыши по строке в NSOutlineView, чтобы вы могли сказать, удалить объект или какое-либо другое действие. (например, когда вы щелкаете правой кнопкой мыши по папке в приложении Apple Mail)

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

@implementation NSContextOutlineView

    - (NSMenu *)defaultMenu {
        if([self selectedRow] < 0) return nil;
        NSMenu *theMenu = [[[NSMenu alloc] initWithTitle:@"Model browser context menu"] autorelease];
        [theMenu insertItemWithTitle:@"Add package" action:@selector(addSite:) keyEquivalent:@"" atIndex:0];
        NSString* deleteItem = [NSString stringWithFormat: @"Remove '%i'", [self selectedRow]];
        [theMenu insertItemWithTitle: deleteItem action:@selector(removeSite:) keyEquivalent:@"" atIndex:1];
        return theMenu;
    }

    - (NSMenu *)menuForEvent:(NSEvent *)theEvent {
        return [self defaultMenu];  
    }
@end

Извините, если ответ очевиден, я просто не могу найти какую-либо помощь по этому вопросу онлайн или в документации.

Спасибо Void за ответ, это привело меня к использованию:

- (NSMenu *)menuForEvent:(NSEvent *)theEvent {
    NSPoint pt = [self convertPoint:[theEvent locationInWindow] fromView:nil];
    id item = [self itemAtRow: [self rowAtPoint:pt]];
    return [self defaultMenuFor: item];
}

Ответы [ 4 ]

23 голосов
/ 21 августа 2009

В вашем методе menuForEvent вы можете узнать, по какой строке произошел щелчок. Вы можете передать это как параметр в defaultMenu метод - возможно, назовите его defaultMenuForRow:

-(NSMenu*)menuForEvent:(NSEvent*)evt 
{
    NSPoint pt = [self convertPoint:[evt locationInWindow] fromView:nil];
    int row=[self rowAtPoint:pt];
    return [self defaultMenuForRow:row];
}

Теперь вы можете построить меню для строки, найденной в событии ...

-(NSMenu*)defaultMenuForRow:(int)row
{
    if (row < 0) return nil;

    NSMenu *theMenu = [[[NSMenu alloc] 
                                initWithTitle:@"Model browser context menu"] 
                                autorelease];
    [theMenu insertItemWithTitle:@"Add package" 
                          action:@selector(addSite:) 
                   keyEquivalent:@"" 
                         atIndex:0];
    [theMenu insertItemWithTitle:[NSString stringWithFormat:@"Remove '%i'", row] 
                          action:@selector(removeSite:) 
                   keyEquivalent:@"" 
                         atIndex:0];
    // you'll need to find a way of getting the information about the 
    // row that is to be removed to the removeSite method
    // assuming that an ivar 'contextRow' is used for this
    contextRow = row;

    return theMenu;        
}

Кроме того, как уже упоминалось в комментариях, вы действительно не должны использовать префикс NS в своих собственных классах. Существует вероятность столкновения в будущем, и это смущает всех, кто смотрит на ваш код, включая вас самих:)

Надеюсь, это поможет ...

12 голосов
/ 22 сентября 2015

Вот пример Swift 2.0, который использует подкласс и расширяет значение по умолчанию NSOutlineDelegate, чтобы вы могли определить свои меню в делегате.

protocol MenuOutlineViewDelegate : NSOutlineViewDelegate {
    func outlineView(outlineView: NSOutlineView, menuForItem item: AnyObject) -> NSMenu?
}

class MenuOutlineView: NSOutlineView {

    override func menuForEvent(event: NSEvent) -> NSMenu? {
        let point = self.convertPoint(event.locationInWindow, fromView: nil)
        let row = self.rowAtPoint(point)
        let item = self.itemAtRow(row)

        if (item == nil) {
            return nil
        }

        return (self.delegate() as! MenuOutlineViewDelegate).outlineView(self, menuForItem: item!)
    }

}
1 голос
/ 13 декабря 2018

Гораздо позже, чем вопрос ОП, но для таких, как я, интересно, вот мое решение. Он также нуждается в создании подкласса NSOutlineView, который Apple в любом случае не поощряет ...

Вместо переопределения menuForEvent: Я переопределяю rightMouseDown:

- (void)rightMouseDown:(NSEvent *)event {
    NSPoint pt = [self convertPoint:[event locationInWindow] fromView:nil];
    NSInteger row = [self rowAtPoint:pt];
    id item = [self itemAtRow:row];
    NSMenu *menu;
    //set the menu to one you have defined either in code or IB through outlets
    self.menu = menu;
    [super rightMouseDown:event];
}

Это имеет преимущество, заключающееся в том, что он сохраняет вызовы делегатов для обновления меню, а также сохраняет выделение строк при щелчке правой кнопкой.

0 голосов
/ 07 декабря 2012

Если вы предпочитаете, вы можете прикрепить меню к отдельному виду ячейки или строке и построить его с помощью построителя интерфейса:

@implementation BSMotleyOutlineView

-(NSMenu *)menuForEvent:(NSEvent *)event
{
    NSPoint pt = [self convertPoint:[event locationInWindow] fromView:nil];
    NSInteger row = [self rowAtPoint:pt];
    if (row >= 0) {
        NSTableRowView* rowView = [self rowViewAtRow:row makeIfNecessary:NO];
        if (rowView) {
            NSInteger col = [self columnAtPoint:pt];
            if (col >= 0) {
                NSTableCellView* cellView = [rowView viewAtColumn:col];
                NSMenu* cellMenu = cellView.menu;
                if(cellMenu) {
                    return cellMenu;
                }
            }
            NSMenu* rowMenu = rowView.menu;
            if (rowMenu) {
                return rowMenu;
            }
        }
    }
    return [super menuForEvent:event];
}
@end
...