/* * Copyright (C) 2006, 2008 Apple Inc. All rights reserved. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public License * along with this library; see the file COPYING.LIB. If not, write to * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301, USA. */ #import "config.h" #import "PopupMenu.h" #import "ChromeClient.h" #import "EventHandler.h" #import "Frame.h" #import "FrameView.h" #import "HTMLNames.h" #import "HTMLOptGroupElement.h" #import "HTMLOptionElement.h" #import "HTMLSelectElement.h" #import "Page.h" #import "SimpleFontData.h" #import "WebCoreSystemInterface.h" namespace WebCore { using namespace HTMLNames; PopupMenu::PopupMenu(PopupMenuClient* client) : m_popupClient(client) { } PopupMenu::~PopupMenu() { if (m_popup) [m_popup.get() setControlView:nil]; } void PopupMenu::clear() { if (m_popup) [m_popup.get() removeAllItems]; } void PopupMenu::populate() { if (m_popup) clear(); else { m_popup = [[NSPopUpButtonCell alloc] initTextCell:@"" pullsDown:!client()->shouldPopOver()]; [m_popup.get() release]; // release here since the RetainPtr has retained the object already [m_popup.get() setUsesItemFromMenu:NO]; [m_popup.get() setAutoenablesItems:NO]; } BOOL messagesEnabled = [[m_popup.get() menu] menuChangedMessagesEnabled]; [[m_popup.get() menu] setMenuChangedMessagesEnabled:NO]; // For pullDown menus the first item is hidden. if (!client()->shouldPopOver()) [m_popup.get() addItemWithTitle:@""]; ASSERT(client()); int size = client()->listSize(); for (int i = 0; i < size; i++) { if (client()->itemIsSeparator(i)) [[m_popup.get() menu] addItem:[NSMenuItem separatorItem]]; else { PopupMenuStyle style = client()->itemStyle(i); NSMutableDictionary* attributes = [[NSMutableDictionary alloc] init]; if (style.font() != Font()) [attributes setObject:style.font().primaryFont()->getNSFont() forKey:NSFontAttributeName]; // FIXME: Add support for styling the foreground and background colors. // FIXME: Find a way to customize text color when an item is highlighted. NSAttributedString* string = [[NSAttributedString alloc] initWithString:client()->itemText(i) attributes:attributes]; [attributes release]; [m_popup.get() addItemWithTitle:@""]; NSMenuItem* menuItem = [m_popup.get() lastItem]; [menuItem setAttributedTitle:string]; [menuItem setEnabled:client()->itemIsEnabled(i)]; [menuItem setToolTip:client()->itemToolTip(i)]; [string release]; } } [[m_popup.get() menu] setMenuChangedMessagesEnabled:messagesEnabled]; } void PopupMenu::show(const IntRect& r, FrameView* v, int index) { populate(); int numItems = [m_popup.get() numberOfItems]; if (numItems <= 0) { if (client()) client()->hidePopup(); return; } ASSERT(numItems > index); // Workaround for crazy bug where a selected index of -1 for a menu with only 1 item will cause a blank menu. if (index == -1 && numItems == 2 && !client()->shouldPopOver() && ![[m_popup.get() itemAtIndex:1] isEnabled]) index = 0; NSView* view = v->documentView(); [m_popup.get() attachPopUpWithFrame:r inView:view]; [m_popup.get() selectItemAtIndex:index]; NSMenu* menu = [m_popup.get() menu]; NSPoint location; NSFont* font = client()->menuStyle().font().primaryFont()->getNSFont(); // These values were borrowed from AppKit to match their placement of the menu. const int popOverHorizontalAdjust = -10; const int popUnderHorizontalAdjust = 6; const int popUnderVerticalAdjust = 6; if (client()->shouldPopOver()) { NSRect titleFrame = [m_popup.get() titleRectForBounds:r]; if (titleFrame.size.width <= 0 || titleFrame.size.height <= 0) titleFrame = r; float vertOffset = roundf((NSMaxY(r) - NSMaxY(titleFrame)) + NSHeight(titleFrame)); // Adjust for fonts other than the system font. NSFont* defaultFont = [NSFont systemFontOfSize:[font pointSize]]; vertOffset += [font descender] - [defaultFont descender]; vertOffset = fminf(NSHeight(r), vertOffset); location = NSMakePoint(NSMinX(r) + popOverHorizontalAdjust, NSMaxY(r) - vertOffset); } else location = NSMakePoint(NSMinX(r) + popUnderHorizontalAdjust, NSMaxY(r) + popUnderVerticalAdjust); // Save the current event that triggered the popup, so we can clean up our event // state after the NSMenu goes away. RefPtr frame = v->frame(); NSEvent* event = [frame->eventHandler()->currentNSEvent() retain]; RefPtr protector(this); RetainPtr dummyView(AdoptNS, [[NSView alloc] initWithFrame:r]); [view addSubview:dummyView.get()]; location = [dummyView.get() convertPoint:location fromView:view]; if (Page* page = frame->page()) page->chrome()->client()->willPopUpMenu(menu); wkPopupMenu(menu, location, roundf(NSWidth(r)), dummyView.get(), index, font); [m_popup.get() dismissPopUp]; [dummyView.get() removeFromSuperview]; if (client()) { int newIndex = [m_popup.get() indexOfSelectedItem]; client()->hidePopup(); // Adjust newIndex for hidden first item. if (!client()->shouldPopOver()) newIndex--; if (index != newIndex && newIndex >= 0) client()->valueChanged(newIndex); // Give the frame a chance to fix up its event state, since the popup eats all the // events during tracking. frame->eventHandler()->sendFakeEventsAfterWidgetTracking(event); } [event release]; } void PopupMenu::hide() { [m_popup.get() dismissPopUp]; } void PopupMenu::updateFromElement() { } bool PopupMenu::itemWritingDirectionIsNatural() { return true; } }