]> git.saurik.com Git - wxWidgets.git/blob - src/unix/taskbarx11.cpp
Fix memory leak (closes bug 1191328)
[wxWidgets.git] / src / unix / taskbarx11.cpp
1 /////////////////////////////////////////////////////////////////////////
2 // File: taskbar.cpp
3 // Purpose: wxTaskBarIcon class for common Unix desktops
4 // Author: Vaclav Slavik
5 // Modified by:
6 // Created: 04/04/2003
7 // RCS-ID: $Id$
8 // Copyright: (c) Vaclav Slavik, 2003
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////
11
12 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA)
13 #pragma implementation "taskbarx11.h"
14 #endif
15
16 // NB: This implementation does *not* work with every X11 window manager.
17 // Currently only GNOME 1.2 and KDE 1,2,3 methods are implemented here.
18 // Freedesktop.org's System Tray specification is implemented in
19 // src/gtk/taskbar.cpp and used from here under wxGTK.
20 //
21 // Thanks to Ian Campbell, author of XMMS Status Docklet, for publishing
22 // KDE and GNOME 1.2 methods.
23
24
25 // For compilers that support precompilation, includes "wx.h".
26 #include "wx/wxprec.h"
27
28 #include "wx/taskbar.h"
29 #include "wx/frame.h"
30 #include "wx/bitmap.h"
31 #include "wx/statbmp.h"
32 #include "wx/sizer.h"
33 #include "wx/dcclient.h"
34 #include "wx/log.h"
35 #include "wx/image.h"
36
37 #ifdef __VMS
38 #pragma message disable nosimpint
39 #endif
40 #include <X11/Xlib.h>
41 #include <X11/Xatom.h>
42 #ifdef __VMS
43 #pragma message enable nosimpint
44 #endif
45
46 // ----------------------------------------------------------------------------
47 // base class that implements toolkit-specific method:
48 // ----------------------------------------------------------------------------
49
50 #ifdef __WXGTK20__
51 #include <gtk/gtk.h>
52 #if GTK_CHECK_VERSION(2,1,0)
53 #include "wx/gtk/taskbarpriv.h"
54 #define TASKBAR_ICON_AREA_BASE_INCLUDED
55 #endif
56 #endif
57
58 #ifndef TASKBAR_ICON_AREA_BASE_INCLUDED
59 class WXDLLIMPEXP_ADV wxTaskBarIconAreaBase : public wxFrame
60 {
61 public:
62 wxTaskBarIconAreaBase()
63 : wxFrame(NULL, wxID_ANY, _T("systray icon"),
64 wxDefaultPosition, wxDefaultSize,
65 wxDEFAULT_FRAME_STYLE | wxFRAME_NO_TASKBAR |
66 wxSIMPLE_BORDER | wxFRAME_SHAPED) {}
67
68 bool IsProtocolSupported() const { return false; }
69 };
70 #endif
71
72
73 // ----------------------------------------------------------------------------
74 // toolkit dependent methods to set properties on helper window:
75 // ----------------------------------------------------------------------------
76
77 #if defined(__WXGTK__)
78 #include <gdk/gdk.h>
79 #include <gdk/gdkx.h>
80 #include <gtk/gtk.h>
81 #define GetDisplay() GDK_DISPLAY()
82 #define GetXWindow(wxwin) GDK_WINDOW_XWINDOW((wxwin)->m_widget->window)
83 #elif defined(__WXX11__) || defined(__WXMOTIF__)
84 #include "wx/x11/privx.h"
85 #define GetDisplay() ((Display*)wxGlobalDisplay())
86 #define GetXWindow(wxwin) ((Window)(wxwin)->GetHandle())
87 #else
88 #error "You must define X11 accessors for this port!"
89 #endif
90
91
92 // ----------------------------------------------------------------------------
93 // wxTaskBarIconArea is the real window that shows the icon:
94 // ----------------------------------------------------------------------------
95
96 class WXDLLIMPEXP_ADV wxTaskBarIconArea : public wxTaskBarIconAreaBase
97 {
98 public:
99 wxTaskBarIconArea(wxTaskBarIcon *icon, const wxBitmap &bmp);
100 void SetTrayIcon(const wxBitmap& bmp);
101 bool IsOk() { return true; }
102
103 protected:
104 void SetLegacyWMProperties();
105
106 void OnSizeChange(wxSizeEvent& event);
107 void OnPaint(wxPaintEvent& evt);
108 void OnMouseEvent(wxMouseEvent& event);
109 void OnMenuEvent(wxCommandEvent& event);
110
111 wxTaskBarIcon *m_icon;
112 wxPoint m_pos;
113 wxBitmap m_bmp;
114
115 DECLARE_EVENT_TABLE()
116 };
117
118 BEGIN_EVENT_TABLE(wxTaskBarIconArea, wxTaskBarIconAreaBase)
119 EVT_SIZE(wxTaskBarIconArea::OnSizeChange)
120 EVT_MOUSE_EVENTS(wxTaskBarIconArea::OnMouseEvent)
121 EVT_MENU(-1, wxTaskBarIconArea::OnMenuEvent)
122 EVT_PAINT(wxTaskBarIconArea::OnPaint)
123 END_EVENT_TABLE()
124
125 wxTaskBarIconArea::wxTaskBarIconArea(wxTaskBarIcon *icon, const wxBitmap &bmp)
126 : wxTaskBarIconAreaBase(), m_icon(icon), m_pos(0,0)
127 {
128 if (!IsProtocolSupported())
129 {
130 wxLogTrace(_T("systray"),
131 _T("using legacy KDE1,2 and GNOME 1.2 methods"));
132 SetLegacyWMProperties();
133 }
134
135 #if defined(__WXGTK20__) && defined(TASKBAR_ICON_AREA_BASE_INCLUDED)
136 m_invokingWindow = icon;
137 #endif
138
139 // Set initial size to bitmap size (tray manager may and often will
140 // change it):
141 SetSize(wxSize(bmp.GetWidth(), bmp.GetHeight()));
142
143 SetTrayIcon(bmp);
144 }
145
146 void wxTaskBarIconArea::SetTrayIcon(const wxBitmap& bmp)
147 {
148 m_bmp = bmp;
149
150 // determine suitable bitmap size:
151 wxSize winsize(GetSize());
152 wxSize bmpsize(m_bmp.GetWidth(), m_bmp.GetHeight());
153 wxSize iconsize(wxMin(winsize.x, bmpsize.x), wxMin(winsize.y, bmpsize.y));
154
155 // rescale the bitmap to fit into the tray icon window:
156 if (bmpsize != iconsize)
157 {
158 wxImage img = m_bmp.ConvertToImage();
159 img.Rescale(iconsize.x, iconsize.y);
160 m_bmp = wxBitmap(img);
161 }
162
163 wxRegion region;
164 region.Union(m_bmp);
165
166 // if the bitmap is smaller than the window, offset it:
167 if (winsize != iconsize)
168 {
169 m_pos.x = (winsize.x - iconsize.x) / 2;
170 m_pos.y = (winsize.y - iconsize.y) / 2;
171 region.Offset(m_pos.x, m_pos.y);
172 }
173
174 // set frame's shape to correct value and redraw:
175 SetShape(region);
176 Refresh();
177 }
178
179 void wxTaskBarIconArea::SetLegacyWMProperties()
180 {
181 #ifdef __WXGTK__
182 gtk_widget_realize(m_widget);
183 #endif
184
185 long data[1];
186
187 // KDE 2 & KDE 3:
188 Atom _KDE_NET_WM_SYSTEM_TRAY_WINDOW_FOR =
189 XInternAtom(GetDisplay(), "_KDE_NET_WM_SYSTEM_TRAY_WINDOW_FOR", False);
190 data[0] = 0;
191 XChangeProperty(GetDisplay(), GetXWindow(this),
192 _KDE_NET_WM_SYSTEM_TRAY_WINDOW_FOR,
193 XA_WINDOW, 32,
194 PropModeReplace, (unsigned char*)data, 1);
195
196 // GNOME 1.2 & KDE 1:
197 Atom KWM_DOCKWINDOW =
198 XInternAtom(GetDisplay(), "KWM_DOCKWINDOW", False);
199 data[0] = 1;
200 XChangeProperty(GetDisplay(), GetXWindow(this),
201 KWM_DOCKWINDOW,
202 KWM_DOCKWINDOW, 32,
203 PropModeReplace, (unsigned char*)data, 1);
204 }
205
206 void wxTaskBarIconArea::OnSizeChange(wxSizeEvent& event)
207 {
208 wxLogTrace(_T("systray"), _T("icon size changed to %i x %i"),
209 GetSize().x, GetSize().y);
210 // rescale or reposition the icon as needed:
211 wxBitmap bmp(m_bmp);
212 SetTrayIcon(bmp);
213 }
214
215 void wxTaskBarIconArea::OnPaint(wxPaintEvent& WXUNUSED(event))
216 {
217 wxPaintDC dc(this);
218 dc.DrawBitmap(m_bmp, m_pos.x, m_pos.y, true);
219 }
220
221 void wxTaskBarIconArea::OnMouseEvent(wxMouseEvent& event)
222 {
223 wxEventType type = 0;
224 wxEventType mtype = event.GetEventType();
225
226 if (mtype == wxEVT_LEFT_DOWN)
227 type = wxEVT_TASKBAR_LEFT_DOWN;
228 else if (mtype == wxEVT_LEFT_UP)
229 type = wxEVT_TASKBAR_LEFT_UP;
230 else if (mtype == wxEVT_LEFT_DCLICK)
231 type = wxEVT_TASKBAR_LEFT_DCLICK;
232 else if (mtype == wxEVT_RIGHT_DOWN)
233 type = wxEVT_TASKBAR_RIGHT_DOWN;
234 else if (mtype == wxEVT_RIGHT_UP)
235 type = wxEVT_TASKBAR_RIGHT_UP;
236 else if (mtype == wxEVT_RIGHT_DCLICK)
237 type = wxEVT_TASKBAR_RIGHT_DCLICK;
238 else if (mtype == wxEVT_MOTION)
239 type = wxEVT_TASKBAR_MOVE;
240 else
241 return;
242
243 wxTaskBarIconEvent e(type, m_icon);
244 m_icon->ProcessEvent(e);
245 }
246
247 void wxTaskBarIconArea::OnMenuEvent(wxCommandEvent& event)
248 {
249 m_icon->ProcessEvent(event);
250 }
251
252 // ----------------------------------------------------------------------------
253 // wxTaskBarIcon class:
254 // ----------------------------------------------------------------------------
255
256 IMPLEMENT_DYNAMIC_CLASS(wxTaskBarIcon, wxEvtHandler)
257
258 wxTaskBarIcon::wxTaskBarIcon() : m_iconWnd(NULL)
259 {
260 }
261
262 wxTaskBarIcon::~wxTaskBarIcon()
263 {
264 if (m_iconWnd)
265 RemoveIcon();
266 }
267
268 bool wxTaskBarIcon::IsOk() const
269 {
270 return true;
271 }
272
273 bool wxTaskBarIcon::IsIconInstalled() const
274 {
275 return m_iconWnd != NULL;
276 }
277
278 bool wxTaskBarIcon::SetIcon(const wxIcon& icon, const wxString& tooltip)
279 {
280 wxBitmap bmp;
281 bmp.CopyFromIcon(icon);
282
283 if (!m_iconWnd)
284 {
285 m_iconWnd = new wxTaskBarIconArea(this, bmp);
286 if (m_iconWnd->IsOk())
287 {
288 m_iconWnd->Show();
289 }
290 else
291 {
292 m_iconWnd->Destroy();
293 m_iconWnd = NULL;
294 return false;
295 }
296 }
297 else
298 {
299 m_iconWnd->SetTrayIcon(bmp);
300 }
301
302 #if wxUSE_TOOLTIPS
303 if (!tooltip.empty())
304 m_iconWnd->SetToolTip(tooltip);
305 else
306 m_iconWnd->SetToolTip(NULL);
307 #endif
308 return true;
309 }
310
311 bool wxTaskBarIcon::RemoveIcon()
312 {
313 if (!m_iconWnd)
314 return false;
315 m_iconWnd->Destroy();
316 m_iconWnd = NULL;
317 return true;
318 }
319
320 bool wxTaskBarIcon::PopupMenu(wxMenu *menu)
321 {
322 if (!m_iconWnd)
323 return false;
324 m_iconWnd->PopupMenu(menu);
325 return true;
326 }