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