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

Всплывающее меню из NSButton

Как мне это сделать?

Я думал о...

[NSMenu popUpContextMenu:menu withEvent:event forView:(NSButton *)sender];
4b9b3361

Ответ 1

Да.

При вызове действия кнопки

[NSMenu popUpContextMenu:menu withEvent:event forView:(NSButton *)sender];

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

Ответ 2

Быстрая версия принятого ответа

@IBAction func actionOccurred(sender: NSButton) {
    if let event = NSApplication.sharedApplication().currentEvent {
        NSMenu.popUpContextMenu(sender.menu!, withEvent: event, forView: sender)
    }
}

Ответ 3

Использование контекстного меню в вызове действий не является отличным способом сделать это, потому что меню не отображается до тех пор, пока mouseUp - вы не получите поведение удержания и перетаскивания меню. Образец Apple ButtonMadness показывает, как это сделать в подклассе NSButton, см. DropDownButton. https://developer.apple.com/library/mac/samplecode/ButtonMadness/Introduction/Intro.html

Подводя итог этому подклассу: создайте NSPopUpButtonCell с pullsDown, установленным в YES и preferredEdge, чтобы NSMaxYEdge, скопируйте свое меню, чтобы добавить чистый верхний элемент и установите его как это меню ячеек, на вызов mouseDown [thePopUpCell performClickWithFrame: self.bounds inView: self ] и установите self.needsDisplay

Ответ 4

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

Честно говоря, NSPopupButton - лучший выбор в большинстве ситуаций. Я использую этот код главным образом из-за удобства наличия одного класса для кнопок и всплывающих окон, а также потому, что в меню нет изображения и заголовка всплывающего окна. Я загружаю меню из отдельного наконечника и повторно использую его, как и в другом месте приложения, по мере необходимости.

Обратите внимание: тривиально добавить дополнительную поддержку, например, как popover, так и меню.

NSButton subclass:

- (void)mouseDown:(NSEvent *)theEvent {

    // if a menu is defined let the cell handle its display
    if (self.menu) {
        if ([theEvent type] == NSLeftMouseDown) {
            [[self cell] setMenu:[self menu]];
        } else {
            [[self cell] setMenu:nil];
        }
    }

    [super mouseDown:theEvent];
}

NSButtonCell subclass:

- (BOOL)trackMouse:(NSEvent *)event inRect:(NSRect)cellFrame ofView:(NSView *)controlView untilMouseUp:(BOOL)untilMouseUp
{
    // if menu defined show on left mouse
    if ([event type] == NSLeftMouseDown && [self menu]) {

        NSPoint result = [controlView convertPoint:NSMakePoint(NSMidX(cellFrame), NSMidY(cellFrame)) toView:nil];

        NSEvent *newEvent = [NSEvent mouseEventWithType: [event type]
                                               location: result
                                          modifierFlags: [event modifierFlags]
                                              timestamp: [event timestamp]
                                           windowNumber: [event windowNumber]
                                                context: [event context]
                                            eventNumber: [event eventNumber]
                                             clickCount: [event clickCount]
                                               pressure: [event pressure]];

        // need to generate a new event otherwise selection of button
        // after menu display fails
        [NSMenu popUpContextMenu:[self menu] withEvent:newEvent forView:controlView];

        return YES;
    }

    return [super trackMouse:event inRect:cellFrame ofView:controlView untilMouseUp:untilMouseUp];
}

Ответ 5

Недавно я пытался реализовать его, и я пришел, как я думаю, с более простым решением

-(IBAction)buttonClick:(id)sender {
    NSButton * b = (NSButton*)sender;
    NSPoint l = [ self.window convertBaseToScreen:b.frame.origin ];

    [ self.menu popUpMenuPositioningItem:nil atLocation:l inView:nil ];
}

Обновление

convertBaseToScreen устарел, начиная с 10.7, вместо него используйте convertRectToScreen следующим образом:

NSPoint l = [self.window convertRectToScreen:b.frame].origin;