Как использовать NSStatusItem в качестве места назначения перетаскивания?

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

@interface CMDroppableView : NSView <NSMenuDelegate>

Я хотел не только принимать операции перетаскивания, но и предоставлять NSMenu, когда пользователь нажимает на значок. Мне удалось заставить NSMenu правильно отображаться, но функциональность перетаскивания остается неуловимой.

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

-(void)awakeFromNib{
[self registerForDraggedTypes:[NSArray arrayWithObject:NSFilenamesPboardType]];
}

Перетащите сообщения:

-(NSDragOperation)draggingEntered:(id <NSDraggingInfo>)sender{
    NSLog(@"Drag Enter");
    return NSDragOperationCopy;
}

-(NSDragOperation)draggingUpdated:(id <NSDraggingInfo>)sender{
    return NSDragOperationCopy;
}

-(void)draggingExited:(id <NSDraggingInfo>)sender{
    NSLog(@"Drag Exit");
}

-(BOOL)prepareForDragOperation:(id <NSDraggingInfo>)sender{
     return YES;
}

-(BOOL)performDragOperation:(id <NSDraggingInfo>)sender{
    return YES;
}

Вот код, в котором установлен пользовательский вид:

statusItemView = [[CMDroppableView alloc] init];
[statusItemView retain];
[statusItemView setMenu: statusMenu];

[statusItem setView: statusItemView];   

Еще ничего. Так где я ошибся?

Спасибо!

2 ответа

Решение

Изменить: D'Oh, вы регистрируете типы перетаскивания в -awakeFromNib, который не будет вызываться, если представление не загружается из пера. Попробуйте зарегистрировать ваши типы перетаскивания в -initWithFrame: вместо!

Старый ответ:

IIRC вам не нужно устанавливать меню на пункт статуса. Что я делаю, так это настраиваю мой вид, управляю меню и делаю что-то вроде этого:

- (void)setMenu:(NSMenu *)menu {
    [menu setDelegate:self];
    [super setMenu:menu];
}

- (void)mouseDown:(NSEvent *)event {
    [statusItem popUpStatusItemMenu:[self menu]]; // or another method that returns a menu
}

- (void)menuWillOpen:(NSMenu *)menu {
    highlight = YES;
    [self setNeedsDisplay:YES];
}

- (void)menuDidClose:(NSMenu *)menu {
    highlight = NO;
    [self setNeedsDisplay:YES];
}

- (void)drawRect:(NSRect)rect {
    NSImage *img = highlight ? [alternateImage copy] : [image copy];
    NSRect bounds = [self bounds];
    [statusItem drawStatusBarBackgroundInRect:bounds withHighlight:highlight];

    // rest of drawing code goes here, including drawing img where appropriate
}

в реализации моего пользовательского представления. Это гарантирует, что поведение меню идентично поведению по умолчанию.

Методом проб и ошибок я наткнулся на решение. Я регистрировал свои приемлемые типы перетаскивания в сообщении awakeFromNib (как видно из исходного вопроса). Однако я не использовал IB для настройки этого представления, поэтому этот метод никогда не вызывался. Перемещение регистрационного кода в сообщение initFromFrame, похоже, решило проблему.

- (id)initWithFrame:(NSRect)frame {
    self = [super initWithFrame:frame];
    if (self) {
        ...
        [self registerForDraggedTypes:[NSArray arrayWithObject:NSFilenamesPboardType]];
    }
    return self;
}
Другие вопросы по тегам