WebCore:
[WebKit-https.git] / WebCore / platform / mac / PopupMenuMac.mm
1 /*
2  * Copyright (C) 2006 Apple Computer, Inc.
3  *
4  * This library is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU Library General Public
6  * License as published by the Free Software Foundation; either
7  * version 2 of the License, or (at your option) any later version.
8  *
9  * This library is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * Library General Public License for more details.
13  *
14  * You should have received a copy of the GNU Library General Public License
15  * along with this library; see the file COPYING.LIB.  If not, write to
16  * the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
17  * Boston, MA 02111-1307, USA.
18  */
19
20 #import "config.h"
21 #import "PopupMenu.h"
22
23 #import "FontData.h"
24 #import "FrameMac.h"
25 #import "FrameView.h"
26 #import "HTMLNames.h"
27 #import "HTMLOptGroupElement.h"
28 #import "HTMLOptionElement.h"
29 #import "RenderMenuList.h"
30 #import "WebCoreSystemInterface.h"
31
32 namespace WebCore {
33
34 using namespace HTMLNames;
35
36 PopupMenu::PopupMenu(RenderMenuList* menuList)
37     : m_menuList(menuList)
38     , popup(nil)
39 {
40 }
41
42 PopupMenu::~PopupMenu()
43 {
44     if (popup) {
45         [popup setControlView:nil];
46         [popup release];
47     }
48 }
49
50 void PopupMenu::clear()
51 {
52     if (popup)
53         [popup removeAllItems];
54 }
55
56 void PopupMenu::populate()
57 {
58     if (popup)
59         [popup removeAllItems];
60     else {
61         popup = [[NSPopUpButtonCell alloc] initTextCell:@"" pullsDown:NO];
62         [popup setUsesItemFromMenu:NO];
63         [popup setAutoenablesItems:NO];
64     }
65     BOOL messagesEnabled = [[popup menu] menuChangedMessagesEnabled];
66     [[popup menu] setMenuChangedMessagesEnabled:NO];
67     PopupMenu::addItems();
68     [[popup menu] setMenuChangedMessagesEnabled:messagesEnabled];
69 }
70
71 void PopupMenu::show(const IntRect& r, FrameView* v, int index)
72 {
73     populate();
74     if ([popup numberOfItems] <= 0)
75         return;
76     ASSERT([popup numberOfItems] > index);
77
78     NSView* view = v->getDocumentView();
79
80     [popup attachPopUpWithFrame:r inView:view];
81     [popup selectItemAtIndex:index];
82     
83     NSFont* font = menuList()->style()->font().primaryFont()->getNSFont();
84
85     NSRect titleFrame = [popup titleRectForBounds:r];
86     if (titleFrame.size.width <= 0 || titleFrame.size.height <= 0)
87         titleFrame = r;
88     float vertOffset = roundf((NSMaxY(r) - NSMaxY(titleFrame)) + NSHeight(titleFrame));
89     // Adjust for fonts other than the system font.
90     NSFont* defaultFont = [NSFont systemFontOfSize:[font pointSize]];
91     vertOffset += [font descender] - [defaultFont descender];
92     vertOffset = fmin(NSHeight(r), vertOffset);
93
94     NSMenu* menu = [popup menu];
95     // FIXME: Need to document where this magic number 10 comes from.
96     NSPoint location = NSMakePoint(NSMinX(r) - 10, NSMaxY(r) - vertOffset);
97
98     // Save the current event that triggered the popup, so we can clean up our event
99     // state after the NSMenu goes away.
100     RefPtr<FrameMac> frame = Mac(v->frame());
101     NSEvent* event = [frame->currentEvent() retain];
102     
103     RefPtr<PopupMenu> protector(this);
104     
105     frame->willPopupMenu(menu);
106     wkPopupMenu(menu, location, roundf(NSWidth(r)), view, index, font);
107
108     if (menuList()) {
109         int newIndex = [popup indexOfSelectedItem];
110         menuList()->hidePopup();
111
112         if (index != newIndex && newIndex >= 0)
113             menuList()->valueChanged(newIndex);
114
115         // Give the frame a chance to fix up its event state, since the popup eats all the
116         // events during tracking.
117         frame->sendFakeEventsAfterWidgetTracking(event);
118     }
119
120     [event release];
121 }
122
123 void PopupMenu::hide()
124 {
125     [popup dismissPopUp];
126 }
127
128 void PopupMenu::addSeparator()
129 {
130     [[popup menu] addItem:[NSMenuItem separatorItem]];
131 }
132
133 void PopupMenu::addGroupLabel(HTMLOptGroupElement* element)
134 {
135     String text = element->groupLabelText();
136
137     RenderStyle* s = element->renderStyle();
138     if (!s)
139         s = menuList()->style();
140
141     NSMutableDictionary* attributes = [[NSMutableDictionary alloc] init];
142     if (s->font() != Font())
143         [attributes setObject:s->font().primaryFont()->getNSFont() forKey:NSFontAttributeName];
144     // FIXME: Add support for styling the foreground and background colors.
145     NSAttributedString* string = [[NSAttributedString alloc] initWithString:text attributes:attributes];
146     [attributes release];
147
148     [popup addItemWithTitle:@""];
149     NSMenuItem* menuItem = [popup lastItem];
150     [menuItem setAttributedTitle:string];
151     [menuItem setEnabled:NO];
152
153     [string release];
154 }
155
156 void PopupMenu::addOption(HTMLOptionElement* element)
157 {
158     String text = element->optionText();
159     
160     bool groupEnabled = true;
161     if (element->parentNode() && element->parentNode()->hasTagName(optgroupTag))
162         groupEnabled = element->parentNode()->isEnabled();
163
164     RenderStyle* s = element->renderStyle();
165     if (!s)
166         s = menuList()->style();
167         
168     NSMutableDictionary* attributes = [[NSMutableDictionary alloc] init];
169     if (s->font() != Font())
170         [attributes setObject:s->font().primaryFont()->getNSFont() forKey:NSFontAttributeName];
171     // FIXME: Add support for styling the foreground and background colors.
172     // FIXME: Find a way to customize text color when an item is highlighted.
173     NSAttributedString* string = [[NSAttributedString alloc] initWithString:text attributes:attributes];
174     [attributes release];
175
176     [popup addItemWithTitle:@""];
177     NSMenuItem* menuItem = [popup lastItem];
178     [menuItem setAttributedTitle:string];
179     [menuItem setEnabled:groupEnabled && element->isEnabled()];
180
181     [string release];
182 }
183
184 }