]> git.saurik.com Git - wxWidgets.git/blob - src/osx/cocoa/taskbar.mm
Add support for task bar icon tooltips to wxOSX version.
[wxWidgets.git] / src / osx / cocoa / taskbar.mm
1 /////////////////////////////////////////////////////////////////////////
2 // File: src/osx/cocoa/taskbar.mm
3 // Purpose: Implements wxTaskBarIcon class
4 // Author: David Elliott, Stefan Csomor
5 // Modified by:
6 // Created: 2004/01/24
7 // RCS-ID: $Id$
8 // Copyright: (c) 2004 David Elliott, Stefan Csomor
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////
11
12 #include "wx/wxprec.h"
13 #if wxUSE_TASKBARICON
14
15 #ifndef WX_PRECOMP
16 #include "wx/toplevel.h"
17 #include "wx/menu.h"
18 #include "wx/icon.h"
19 #include "wx/log.h"
20 #include "wx/dcclient.h"
21 #endif
22
23 #include "wx/taskbar.h"
24
25 #include "wx/osx/private.h"
26
27 class wxTaskBarIconWindow;
28
29 //-----------------------------------------------------------------------------
30 //
31 // wxTaskBarIconWindow
32 //
33 // Event handler for menus
34 // NB: Since wxWindows in Mac HAVE to have parents we need this to be
35 // a top level window...
36 //-----------------------------------------------------------------------------
37
38 class wxTaskBarIconWindow : public wxTopLevelWindow
39 {
40 public:
41 wxTaskBarIconWindow(wxTaskBarIconImpl *impl);
42
43 void OnMenuEvent(wxCommandEvent& event);
44 void OnUpdateUIEvent(wxUpdateUIEvent& event);
45
46 private:
47 wxTaskBarIconImpl *m_impl;
48 DECLARE_EVENT_TABLE()
49 };
50
51 // ============================================================================
52 // wxTaskBarIconImpl
53 // Base class for the various Cocoa implementations.
54 // ============================================================================
55 class wxTaskBarIconImpl
56 {
57 public:
58 wxTaskBarIconImpl(wxTaskBarIcon *taskBarIcon);
59
60 virtual bool IsStatusItem() const { return false; }
61
62 virtual bool SetIcon(const wxIcon& icon, const wxString& tooltip = wxEmptyString) = 0;
63 virtual bool RemoveIcon() = 0;
64
65 bool IsIconInstalled() const { return m_icon.IsOk(); }
66
67 virtual bool PopupMenu(wxMenu *menu) = 0;
68 virtual ~wxTaskBarIconImpl();
69 inline wxTaskBarIcon* GetTaskBarIcon() { return m_taskBarIcon; }
70 wxMenu * CreatePopupMenu()
71 { return m_taskBarIcon->CreatePopupMenu(); }
72
73 wxDECLARE_NO_COPY_CLASS(wxTaskBarIconImpl);
74
75 protected:
76 wxTaskBarIcon *m_taskBarIcon;
77 wxBitmap m_icon;
78 wxTaskBarIconWindow *m_eventWindow;
79 private:
80 wxTaskBarIconImpl();
81 };
82
83 // ============================================================================
84 // wxTaskBarIconDockImpl
85 // An implementation using the Dock icon.
86 // ============================================================================
87 class wxTaskBarIconDockImpl: public wxTaskBarIconImpl
88 {
89 public:
90 wxTaskBarIconDockImpl(wxTaskBarIcon *taskBarIcon);
91 virtual ~wxTaskBarIconDockImpl();
92 virtual bool SetIcon(const wxIcon& icon, const wxString& tooltip = wxEmptyString);
93 virtual bool RemoveIcon();
94 virtual bool PopupMenu(wxMenu *menu);
95
96 static WX_NSMenu OSXGetDockHMenu();
97 protected:
98 WX_NSMenu OSXDoGetDockHMenu();
99 // There can be only one Dock icon, so make sure we keep it that way
100 static wxTaskBarIconDockImpl *sm_dockIcon;
101 private:
102 wxTaskBarIconDockImpl();
103 wxMenu *m_pMenu;
104 };
105
106 class wxTaskBarIconCustomStatusItemImpl;
107
108 @interface wxOSXStatusItemTarget : NSObject
109 {
110 wxTaskBarIconCustomStatusItemImpl* impl;
111 }
112 @end
113
114 // ============================================================================
115 // wxTaskBarIconCustomStatusItemImpl
116 // An implementation using an NSStatusItem with a custom NSView
117 // ============================================================================
118 class wxTaskBarIconCustomStatusItemImpl: public wxTaskBarIconImpl
119 {
120 public:
121 wxTaskBarIconCustomStatusItemImpl(wxTaskBarIcon *taskBarIcon);
122 virtual ~wxTaskBarIconCustomStatusItemImpl();
123
124 virtual bool IsStatusItem() const { return true; }
125
126 virtual bool SetIcon(const wxIcon& icon, const wxString& tooltip = wxEmptyString);
127 virtual bool RemoveIcon();
128 virtual bool PopupMenu(wxMenu *menu);
129 protected:
130 NSStatusItem *m_statusItem;
131 wxOSXStatusItemTarget *m_target;
132 private:
133 wxTaskBarIconCustomStatusItemImpl();
134 };
135
136 // ============================================================================
137 // wxTaskBarIcon implementation
138 // The facade class.
139 // ============================================================================
140 IMPLEMENT_DYNAMIC_CLASS(wxTaskBarIcon, wxEvtHandler)
141
142 wxTaskBarIcon::wxTaskBarIcon(wxTaskBarIconType iconType)
143 {
144 if(iconType == wxTBI_DOCK)
145 m_impl = new wxTaskBarIconDockImpl(this);
146 else if(iconType == wxTBI_CUSTOM_STATUSITEM)
147 m_impl = new wxTaskBarIconCustomStatusItemImpl(this);
148 else
149 { m_impl = NULL;
150 wxFAIL_MSG(wxT("Invalid wxTaskBarIcon type"));
151 }
152 }
153
154 wxTaskBarIcon::~wxTaskBarIcon()
155 {
156 if ( m_impl )
157 {
158 if ( m_impl->IsIconInstalled() )
159 m_impl->RemoveIcon();
160 delete m_impl;
161 m_impl = NULL;
162 }
163 }
164
165 bool wxTaskBarIcon::OSXIsStatusItem()
166 {
167 if ( m_impl )
168 return m_impl->IsStatusItem();
169
170 return false;
171 }
172
173 // Operations
174
175 bool wxTaskBarIcon::IsIconInstalled() const
176 {
177 if ( m_impl )
178 return m_impl->IsIconInstalled();
179
180 return false;
181 }
182
183 bool wxTaskBarIcon::SetIcon(const wxIcon& icon, const wxString& tooltip)
184 {
185 if ( m_impl )
186 return m_impl->SetIcon(icon,tooltip);
187
188 return false;
189 }
190
191 bool wxTaskBarIcon::RemoveIcon()
192 {
193 if ( m_impl )
194 return m_impl->RemoveIcon();
195
196 return false;
197 }
198
199 bool wxTaskBarIcon::PopupMenu(wxMenu *menu)
200 {
201 if ( m_impl )
202 return m_impl->PopupMenu(menu);
203
204 return false;
205 }
206
207 // ============================================================================
208 // wxTaskBarIconImpl
209 // ============================================================================
210
211 wxTaskBarIconImpl::wxTaskBarIconImpl(wxTaskBarIcon* taskBarIcon)
212 : m_taskBarIcon(taskBarIcon), m_eventWindow(new wxTaskBarIconWindow(this))
213 {
214 }
215
216 wxTaskBarIconImpl::~wxTaskBarIconImpl()
217 {
218 delete m_eventWindow;
219 }
220
221 // ============================================================================
222 // wxTaskBarIconDockImpl
223 // ============================================================================
224 wxTaskBarIconDockImpl *wxTaskBarIconDockImpl::sm_dockIcon = NULL;
225
226 wxTaskBarIconDockImpl::wxTaskBarIconDockImpl(wxTaskBarIcon *taskBarIcon)
227 : wxTaskBarIconImpl(taskBarIcon)
228 {
229 wxASSERT_MSG(!sm_dockIcon, wxT("You should never have more than one dock icon!"));
230 sm_dockIcon = this;
231 m_pMenu = NULL;
232 }
233
234 wxTaskBarIconDockImpl::~wxTaskBarIconDockImpl()
235 {
236 if(sm_dockIcon == this)
237 sm_dockIcon = NULL;
238 }
239
240 WX_NSMenu wxTaskBarIconDockImpl::OSXGetDockHMenu()
241 {
242 if(sm_dockIcon)
243 return sm_dockIcon->OSXDoGetDockHMenu();
244
245 return nil;
246 }
247
248 WX_NSMenu wxTaskBarIconDockImpl::OSXDoGetDockHMenu()
249 {
250 wxMenu *dockMenu = CreatePopupMenu();
251
252 if(!dockMenu)
253 return nil;
254
255 wxDELETE(m_pMenu);
256
257 m_pMenu = dockMenu;
258
259 m_pMenu->SetInvokingWindow(m_eventWindow);
260
261 m_pMenu->UpdateUI();
262
263 return (WX_NSMenu)dockMenu->GetHMenu();
264 }
265
266 bool wxTaskBarIconDockImpl::SetIcon(const wxIcon& icon, const wxString& WXUNUSED(tooltip))
267 {
268 m_icon.CopyFromIcon(icon);
269 [[NSApplication sharedApplication] setApplicationIconImage:m_icon.GetNSImage()];
270 return true;
271 }
272
273 bool wxTaskBarIconDockImpl::RemoveIcon()
274 {
275 wxDELETE(m_pMenu);
276 m_icon = wxBitmap();
277 [[NSApplication sharedApplication] setApplicationIconImage:nil];
278 return true;
279 }
280
281 bool wxTaskBarIconDockImpl::PopupMenu(wxMenu *WXUNUSED(menu))
282 {
283 wxFAIL_MSG(wxT("You cannot force the Dock icon menu to popup"));
284 return false;
285 }
286
287 @interface wxNSAppController(wxTaskBarIconNSApplicationDelegateCategory)
288 - (NSMenu*)applicationDockMenu:(NSApplication *)sender;
289 @end
290
291 @implementation wxNSAppController(wxTaskBarIconNSApplicationDelegateCategory)
292 - (NSMenu*)applicationDockMenu:(NSApplication *)sender
293 {
294 wxUnusedVar(sender);
295
296 return wxTaskBarIconDockImpl::OSXGetDockHMenu();
297 }
298 @end
299
300 // ============================================================================
301 // wxTaskBarIconCustomStatusItemImpl
302 // ============================================================================
303
304 @implementation wxOSXStatusItemTarget
305
306 - (void) clickedAction: (id) sender
307 {
308 wxUnusedVar(sender);
309 wxMenu *menu = impl->CreatePopupMenu();
310 if (menu)
311 {
312 impl->PopupMenu(menu);
313 delete menu;
314 }
315 }
316
317 - (void)setImplementation: (wxTaskBarIconCustomStatusItemImpl *) theImplementation
318 {
319 impl = theImplementation;
320 }
321
322 - (wxTaskBarIconCustomStatusItemImpl*) implementation
323 {
324 return impl;
325 }
326
327 @end
328
329
330 wxTaskBarIconCustomStatusItemImpl::wxTaskBarIconCustomStatusItemImpl(wxTaskBarIcon *taskBarIcon)
331 : wxTaskBarIconImpl(taskBarIcon)
332 {
333 m_statusItem = nil;
334 m_target = nil;
335 }
336
337 wxTaskBarIconCustomStatusItemImpl::~wxTaskBarIconCustomStatusItemImpl()
338 {
339 }
340
341 bool wxTaskBarIconCustomStatusItemImpl::SetIcon(const wxIcon& icon, const wxString& tooltip)
342 {
343 if(!m_statusItem)
344 {
345 m_statusItem = [[NSStatusBar systemStatusBar] statusItemWithLength:NSVariableStatusItemLength];
346 [m_statusItem retain];
347
348 m_target = [[wxOSXStatusItemTarget alloc] init];
349 [m_target setImplementation:this];
350 [m_statusItem setHighlightMode:YES];
351 [m_statusItem setTarget:m_target];
352 [m_statusItem setAction:@selector(clickedAction:)];
353 [m_statusItem sendActionOn:NSLeftMouseDownMask];
354 }
355
356 m_icon.CopyFromIcon(icon);
357
358 // status item doesn't scale automatically
359
360 int dimension = m_icon.GetHeight();
361 if ( m_icon.GetWidth() > dimension )
362 dimension = m_icon.GetWidth();
363 if ( dimension > 16 )
364 {
365 wxImage img = m_icon.ConvertToImage();
366 int factor = (dimension+15)/16;
367 m_icon = img.ShrinkBy(factor, factor);
368 }
369
370 [m_statusItem setImage:m_icon.GetNSImage()];
371 wxCFStringRef cfTooltip(tooltip);
372 [m_statusItem setToolTip:cfTooltip.AsNSString()];
373 return true;
374 }
375
376 bool wxTaskBarIconCustomStatusItemImpl::RemoveIcon()
377 {
378 [m_statusItem release];
379 m_statusItem = nil;
380 [m_target release];
381 m_target = nil;
382
383 m_icon = wxBitmap();
384
385 return true;
386 }
387
388 bool wxTaskBarIconCustomStatusItemImpl::PopupMenu(wxMenu *menu)
389 {
390 wxASSERT(menu);
391
392 menu->SetInvokingWindow(m_eventWindow);
393 menu->UpdateUI();
394
395 [m_statusItem popUpStatusItemMenu:(NSMenu*)menu->GetHMenu()];
396
397 menu->SetInvokingWindow(NULL);
398 return true;
399 }
400
401 // ============================================================================
402 // wxTaskBarIconWindow
403 // ============================================================================
404
405 BEGIN_EVENT_TABLE(wxTaskBarIconWindow, wxWindow)
406 EVT_MENU(-1, wxTaskBarIconWindow::OnMenuEvent)
407 EVT_UPDATE_UI(-1, wxTaskBarIconWindow::OnUpdateUIEvent)
408 END_EVENT_TABLE()
409
410 wxTaskBarIconWindow::wxTaskBarIconWindow(wxTaskBarIconImpl *impl)
411 : m_impl(impl)
412 {
413 }
414
415 void wxTaskBarIconWindow::OnMenuEvent(wxCommandEvent& event)
416 {
417 m_impl->GetTaskBarIcon()->ProcessEvent(event);
418 }
419
420 void wxTaskBarIconWindow::OnUpdateUIEvent(wxUpdateUIEvent& event)
421 {
422 m_impl->GetTaskBarIcon()->ProcessEvent(event);
423 }
424
425 #endif //def wxHAS_TASK_BAR_ICON