47
|
1 /*******************************************************************************
|
|
2 * Copyright (c) 2000, 2007 IBM Corporation and others.
|
|
3 * All rights reserved. This program and the accompanying materials
|
|
4 * are made available under the terms of the Eclipse Public License v1.0
|
|
5 * which accompanies this distribution, and is available at
|
|
6 * http://www.eclipse.org/legal/epl-v10.html
|
|
7 *
|
|
8 * Contributors:
|
|
9 * IBM Corporation - initial API and implementation
|
|
10 *******************************************************************************/
|
|
11 module dwt.widgets.ToolTip;
|
|
12
|
|
13 import dwt.widgets.Widget;
|
|
14 import dwt.widgets.TrayItem;
|
|
15 import dwt.widgets.Shell;
|
|
16 import dwt.SWT;
|
|
17 //import dwt.internal.*;
|
|
18 import dwt.internal.gtk.OS;
|
|
19 import dwt.graphics.Point;
|
|
20 import dwt.graphics.Color;
|
|
21 import dwt.events.SelectionListener;
|
|
22 import dwt.events.SelectionEvent;
|
|
23 import dwt.widgets.TypedListener;
|
|
24 import dwt.widgets.Event;
|
|
25 import dwt.widgets.Display;
|
|
26
|
|
27 import tango.stdc.stringz;
|
|
28
|
|
29 /**
|
|
30 * Instances of this class represent popup windows that are used
|
|
31 * to inform or warn the user.
|
|
32 * <p>
|
|
33 * <dl>
|
|
34 * <dt><b>Styles:</b></dt>
|
|
35 * <dd>BALLOON, ICON_ERROR, ICON_INFORMATION, ICON_WARNING</dd>
|
|
36 * <dt><b>Events:</b></dt>
|
|
37 * <dd>Selection</dd>
|
|
38 * </dl>
|
|
39 * </p><p>
|
|
40 * Note: Only one of the styles ICON_ERROR, ICON_INFORMATION,
|
|
41 * and ICON_WARNING may be specified.
|
|
42 * </p><p>
|
|
43 * IMPORTANT: This class is intended to be subclassed <em>only</em>
|
|
44 * within the SWT implementation.
|
|
45 * </p>
|
|
46 *
|
|
47 * @since 3.2
|
|
48 */
|
|
49 public class ToolTip : Widget {
|
|
50 Shell parent;
|
|
51 char[] text, message;
|
|
52 TrayItem item;
|
|
53 int x, y, timerId;
|
|
54 void* layoutText, layoutMessage;
|
|
55 int [] borderPolygon;
|
|
56 bool spikeAbove, autohide;
|
|
57 CallbackData timerProcCallbackData;
|
|
58
|
|
59 static final int BORDER = 5;
|
|
60 static final int PADDING = 5;
|
|
61 static final int INSET = 4;
|
|
62 static final int TIP_HEIGHT = 20;
|
|
63 static final int IMAGE_SIZE = 16;
|
|
64 static final int DELAY = 8000;
|
|
65
|
|
66 /**
|
|
67 * Constructs a new instance of this class given its parent
|
|
68 * and a style value describing its behavior and appearance.
|
|
69 * <p>
|
|
70 * The style value is either one of the style constants defined in
|
|
71 * class <code>SWT</code> which is applicable to instances of this
|
|
72 * class, or must be built by <em>bitwise OR</em>'ing together
|
|
73 * (that is, using the <code>int</code> "|" operator) two or more
|
|
74 * of those <code>SWT</code> style constants. The class description
|
|
75 * lists the style constants that are applicable to the class.
|
|
76 * Style bits are also inherited from superclasses.
|
|
77 * </p>
|
|
78 *
|
|
79 * @param parent a composite control which will be the parent of the new instance (cannot be null)
|
|
80 * @param style the style of control to construct
|
|
81 *
|
|
82 * @exception IllegalArgumentException <ul>
|
|
83 * <li>ERROR_NULL_ARGUMENT - if the parent is null</li>
|
|
84 * </ul>
|
|
85 * @exception SWTException <ul>
|
|
86 * <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the parent</li>
|
|
87 * <li>ERROR_INVALID_SUBCLASS - if this class is not an allowed subclass</li>
|
|
88 * </ul>
|
|
89 *
|
|
90 * @see SWT#ICON_ERROR
|
|
91 * @see SWT#ICON_INFORMATION
|
|
92 * @see SWT#ICON_WARNING
|
|
93 * @see Widget#checkSubclass
|
|
94 * @see Widget#getStyle
|
|
95 */
|
|
96 public this (Shell parent, int style) {
|
|
97 super (parent, checkStyle (style));
|
|
98 this.parent = parent;
|
|
99 createWidget (0);
|
|
100 }
|
|
101
|
|
102 static int checkStyle (int style) {
|
|
103 int mask = SWT.ICON_ERROR | SWT.ICON_INFORMATION | SWT.ICON_WARNING;
|
|
104 if ((style & mask) is 0) return style;
|
|
105 return checkBits (style, SWT.ICON_INFORMATION, SWT.ICON_WARNING, SWT.ICON_ERROR, 0, 0, 0);
|
|
106 }
|
|
107
|
|
108 /**
|
|
109 * Adds the listener to the collection of listeners who will
|
|
110 * be notified when the receiver is selected by the user, by sending
|
|
111 * it one of the messages defined in the <code>SelectionListener</code>
|
|
112 * interface.
|
|
113 * <p>
|
|
114 * <code>widgetSelected</code> is called when the receiver is selected.
|
|
115 * <code>widgetDefaultSelected</code> is not called.
|
|
116 * </p>
|
|
117 *
|
|
118 * @param listener the listener which should be notified when the receiver is selected by the user
|
|
119 *
|
|
120 * @exception IllegalArgumentException <ul>
|
|
121 * <li>ERROR_NULL_ARGUMENT - if the listener is null</li>
|
|
122 * </ul>
|
|
123 * @exception SWTException <ul>
|
|
124 * <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
|
|
125 * <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
|
|
126 * </ul>
|
|
127 *
|
|
128 * @see SelectionListener
|
|
129 * @see #removeSelectionListener
|
|
130 * @see SelectionEvent
|
|
131 */
|
|
132 public void addSelectionListener (SelectionListener listener) {
|
|
133 checkWidget ();
|
|
134 if (listener is null) error (SWT.ERROR_NULL_ARGUMENT);
|
|
135 TypedListener typedListener = new TypedListener (listener);
|
|
136 addListener (SWT.Selection,typedListener);
|
|
137 addListener (SWT.DefaultSelection,typedListener);
|
|
138 }
|
|
139
|
|
140 void configure () {
|
|
141 auto screen = OS.gdk_screen_get_default ();
|
|
142 OS.gtk_widget_realize (handle);
|
|
143 int monitorNumber = OS.gdk_screen_get_monitor_at_window (screen, OS.GTK_WIDGET_WINDOW (handle));
|
|
144 GdkRectangle* dest = new GdkRectangle ();
|
|
145 OS.gdk_screen_get_monitor_geometry (screen, monitorNumber, dest);
|
|
146 Point point = getSize (dest.width / 4);
|
|
147 int w = point.x;
|
|
148 int h = point.y;
|
|
149 point = getLocation ();
|
|
150 int x = point.x;
|
|
151 int y = point.y;
|
|
152 OS.gtk_window_resize (cast(GtkWindow*)handle, w, h + TIP_HEIGHT);
|
|
153 int[] polyline;
|
|
154 spikeAbove = dest.height >= y + h + TIP_HEIGHT;
|
|
155 if (dest.width >= x + w) {
|
|
156 if (dest.height >= y + h + TIP_HEIGHT) {
|
|
157 int t = TIP_HEIGHT;
|
|
158 polyline = [
|
|
159 0, 5+t, 1, 5+t, 1, 3+t, 3, 1+t, 5, 1+t, 5, t,
|
|
160 16, t, 16, 0, 35, t,
|
|
161 w-5, t, w-5, 1+t, w-3, 1+t, w-1, 3+t, w-1, 5+t, w, 5+t,
|
|
162 w, h-5+t, w-1, h-5+t, w-1, h-3+t, w-2, h-3+t, w-2, h-2+t, w-3, h-2+t, w-3, h-1+t, w-5, h-1+t, w-5, h+t,
|
|
163 5, h+t, 5, h-1+t, 3, h-1+t, 3, h-2+t, 2, h-2+t, 2, h-3+t, 1, h-3+t, 1, h-5+t, 0, h-5+t,
|
|
164 0, 5+t];
|
|
165 borderPolygon = [
|
|
166 0, 5+t, 1, 4+t, 1, 3+t, 3, 1+t, 4, 1+t, 5, t,
|
|
167 16, t, 16, 1, 35, t,
|
|
168 w-6, 0+t, w-5, 1+t, w-4, 1+t, w-2, 3+t, w-2, 4+t, w-1, 5+t,
|
|
169 w-1, h-6+t, w-2, h-5+t, w-2, h-4+t, w-4, h-2+t, w-5, h-2+t, w-6, h-1+t,
|
|
170 5, h-1+t, 4, h-2+t, 3, h-2+t, 1, h-4+t, 1, h-5+t, 0, h-6+t,
|
|
171 0, 5+t];
|
|
172 OS.gtk_window_move (cast(GtkWindow*)handle, Math.max(0, x - 17), y);
|
|
173 } else {
|
|
174 polyline = [
|
|
175 0, 5, 1, 5, 1, 3, 3, 1, 5, 1, 5, 0,
|
|
176 w-5, 0, w-5, 1, w-3, 1, w-1, 3, w-1, 5, w, 5,
|
|
177 w, h-5, w-1, h-5, w-1, h-3, w-2, h-3, w-2, h-2, w-3, h-2, w-3, h-1, w-5, h-1, w-5, h,
|
|
178 35, h, 16, h+TIP_HEIGHT, 16, h,
|
|
179 5, h, 5, h-1, 3, h-1, 3, h-2, 2, h-2, 2, h-3, 1, h-3, 1, h-5, 0, h-5,
|
|
180 0, 5];
|
|
181 borderPolygon = [
|
|
182 0, 5, 1, 4, 1, 3, 3, 1, 4, 1, 5, 0,
|
|
183 w-6, 0, w-5, 1, w-4, 1, w-2, 3, w-2, 4, w-1, 5,
|
|
184 w-1, h-6, w-2, h-5, w-2, h-4, w-4, h-2, w-5, h-2, w-6, h-1,
|
|
185 35, h-1, 17, h+TIP_HEIGHT-2, 17, h-1,
|
|
186 5, h-1, 4, h-2, 3, h-2, 1, h-4, 1, h-5, 0, h-6,
|
|
187 0, 5];
|
|
188 OS.gtk_window_move (cast(GtkWindow*)handle, Math.max(0, x - 17), y - h - TIP_HEIGHT);
|
|
189 }
|
|
190 } else {
|
|
191 if (dest.height >= y + h + TIP_HEIGHT) {
|
|
192 int t = TIP_HEIGHT;
|
|
193 polyline = [
|
|
194 0, 5+t, 1, 5+t, 1, 3+t, 3, 1+t, 5, 1+t, 5, t,
|
|
195 w-35, t, w-16, 0, w-16, t,
|
|
196 w-5, t, w-5, 1+t, w-3, 1+t, w-1, 3+t, w-1, 5+t, w, 5+t,
|
|
197 w, h-5+t, w-1, h-5+t, w-1, h-3+t, w-2, h-3+t, w-2, h-2+t, w-3, h-2+t, w-3, h-1+t, w-5, h-1+t, w-5, h+t,
|
|
198 5, h+t, 5, h-1+t, 3, h-1+t, 3, h-2+t, 2, h-2+t, 2, h-3+t, 1, h-3+t, 1, h-5+t, 0, h-5+t,
|
|
199 0, 5+t];
|
|
200 borderPolygon = [
|
|
201 0, 5+t, 1, 4+t, 1, 3+t, 3, 1+t, 4, 1+t, 5, t,
|
|
202 w-35, t, w-17, 2, w-17, t,
|
|
203 w-6, t, w-5, 1+t, w-4, 1+t, w-2, 3+t, w-2, 4+t, w-1, 5+t,
|
|
204 w-1, h-6+t, w-2, h-5+t, w-2, h-4+t, w-4, h-2+t, w-5, h-2+t, w-6, h-1+t,
|
|
205 5, h-1+t, 4, h-2+t, 3, h-2+t, 1, h-4+t, 1, h-5+t, 0, h-6+t,
|
|
206 0, 5+t];
|
|
207 OS.gtk_window_move (cast(GtkWindow*)handle, Math.min(dest.width - w, x - w + 17), y);
|
|
208 } else {
|
|
209 polyline = [
|
|
210 0, 5, 1, 5, 1, 3, 3, 1, 5, 1, 5, 0,
|
|
211 w-5, 0, w-5, 1, w-3, 1, w-1, 3, w-1, 5, w, 5,
|
|
212 w, h-5, w-1, h-5, w-1, h-3, w-2, h-3, w-2, h-2, w-3, h-2, w-3, h-1, w-5, h-1, w-5, h,
|
|
213 w-16, h, w-16, h+TIP_HEIGHT, w-35, h,
|
|
214 5, h, 5, h-1, 3, h-1, 3, h-2, 2, h-2, 2, h-3, 1, h-3, 1, h-5, 0, h-5,
|
|
215 0, 5];
|
|
216 borderPolygon = [
|
|
217 0, 5, 1, 4, 1, 3, 3, 1, 4, 1, 5, 0,
|
|
218 w-6, 0, w-5, 1, w-4, 1, w-2, 3, w-2, 4, w-1, 5,
|
|
219 w-1, h-6, w-2, h-5, w-2, h-4, w-4, h-2, w-5, h-2, w-6, h-1,
|
|
220 w-17, h-1, w-17, h+TIP_HEIGHT-2, w-36, h-1,
|
|
221 5, h-1, 4, h-2, 3, h-2, 1, h-4, 1, h-5, 0, h-6,
|
|
222 0, 5];
|
|
223 OS.gtk_window_move (cast(GtkWindow*)handle, Math.min(dest.width - w, x - w + 17), y - h - TIP_HEIGHT);
|
|
224 }
|
|
225 }
|
|
226 auto rgn = OS.gdk_region_polygon ( cast(GdkPoint*)polyline.ptr, polyline.length / 2, OS.GDK_EVEN_ODD_RULE);
|
|
227 OS.gtk_widget_realize (handle);
|
|
228 auto window = OS.GTK_WIDGET_WINDOW (handle);
|
|
229 OS.gdk_window_shape_combine_region (window, rgn, 0, 0);
|
|
230 OS.gdk_region_destroy (rgn);
|
|
231 }
|
|
232
|
|
233 void createHandle (int index) {
|
|
234 state |= HANDLE;
|
|
235 if ((style & SWT.BALLOON) !is 0) {
|
|
236 handle = OS.gtk_window_new (OS.GTK_WINDOW_POPUP);
|
|
237 Color background = display.getSystemColor (SWT.COLOR_INFO_BACKGROUND);
|
|
238 OS.gtk_widget_modify_bg (handle, OS.GTK_STATE_NORMAL, background.handle);
|
|
239 OS.gtk_widget_set_app_paintable (handle, true);
|
|
240 } else {
|
|
241 handle = cast(GtkWidget*)OS.gtk_tooltips_new ();
|
|
242 if (handle is null) SWT.error (SWT.ERROR_NO_HANDLES);
|
|
243 /*
|
|
244 * Bug in Solaris-GTK. Invoking gtk_tooltips_force_window()
|
|
245 * can cause a crash in older versions of GTK. The fix is
|
|
246 * to avoid this call if the GTK version is older than 2.2.x.
|
|
247 */
|
|
248 if (OS.GTK_VERSION >= OS.buildVERSION (2, 2, 1)) {
|
|
249 OS.gtk_tooltips_force_window (cast(GtkTooltips*)handle);
|
|
250 }
|
|
251 OS.g_object_ref (handle);
|
|
252 OS.gtk_object_sink (cast(GtkObject*)handle);
|
|
253 }
|
|
254 }
|
|
255
|
|
256 void createWidget (int index) {
|
|
257 super.createWidget (index);
|
|
258 text = "";
|
|
259 message = "";
|
|
260 x = y = -1;
|
|
261 autohide = true;
|
|
262 }
|
|
263
|
|
264 void deregister () {
|
|
265 super.deregister ();
|
|
266 if ((style & SWT.BALLOON) is 0) {
|
|
267 auto tipWindow = OS.GTK_TOOLTIPS_TIP_WINDOW (cast(GtkTooltips*)handle);
|
|
268 if (tipWindow !is null) display.removeWidget (tipWindow);
|
|
269 }
|
|
270 }
|
|
271
|
|
272 void destroyWidget () {
|
|
273 auto topHandle = topHandle ();
|
|
274 releaseHandle ();
|
|
275 if (topHandle !is null && (state & HANDLE) !is 0) {
|
|
276 if ((style & SWT.BALLOON) !is 0) {
|
|
277 OS.gtk_widget_destroy (topHandle);
|
|
278 } else {
|
|
279 OS.g_object_unref (topHandle);
|
|
280 }
|
|
281 }
|
|
282 }
|
|
283
|
|
284 /**
|
|
285 * Returns <code>true</code> if the receiver is automatically
|
|
286 * hidden by the platform, and <code>false</code> otherwise.
|
|
287 *
|
|
288 * @return the receiver's auto hide state
|
|
289 *
|
|
290 * @exception SWTException <ul>
|
|
291 * <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
|
|
292 * <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
|
|
293 * </ul>
|
|
294 *
|
|
295 */
|
|
296 public bool getAutoHide () {
|
|
297 checkWidget ();
|
|
298 return autohide;
|
|
299 }
|
|
300
|
|
301 Point getLocation () {
|
|
302 int x = this.x;
|
|
303 int y = this.y;
|
|
304 if (item !is null) {
|
|
305 auto itemHandle = item.handle;
|
|
306 OS.gtk_widget_realize (itemHandle);
|
|
307 auto window = OS.GTK_WIDGET_WINDOW (itemHandle);
|
|
308 int px, py;
|
|
309 OS.gdk_window_get_origin (cast(GdkWindow*)window, &px, &py);
|
|
310 x = px + OS.GTK_WIDGET_WIDTH (itemHandle) / 2;
|
|
311 y = py + OS.GTK_WIDGET_HEIGHT (itemHandle) / 2;
|
|
312 }
|
|
313 if (x is -1 || y is -1) {
|
|
314 int px, py;
|
|
315 OS.gdk_window_get_pointer (null, &px, &py, null);
|
|
316 x = px;
|
|
317 y = py;
|
|
318 }
|
|
319 return new Point(x, y);
|
|
320 }
|
|
321
|
|
322 /**
|
|
323 * Returns the receiver's message, which will be an empty
|
|
324 * string if it has never been set.
|
|
325 *
|
|
326 * @return the receiver's message
|
|
327 *
|
|
328 * @exception SWTException <ul>
|
|
329 * <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
|
|
330 * <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
|
|
331 * </ul>
|
|
332 */
|
|
333 public char[] getMessage () {
|
|
334 checkWidget ();
|
|
335 return message;
|
|
336 }
|
|
337
|
|
338 char[] getNameText () {
|
|
339 return getText ();
|
|
340 }
|
|
341
|
|
342 /**
|
|
343 * Returns the receiver's parent, which must be a <code>Shell</code>.
|
|
344 *
|
|
345 * @return the receiver's parent
|
|
346 *
|
|
347 * @exception SWTException <ul>
|
|
348 * <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
|
|
349 * <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
|
|
350 * </ul>
|
|
351 */
|
|
352 public Shell getParent () {
|
|
353 checkWidget ();
|
|
354 return parent;
|
|
355 }
|
|
356
|
|
357 Point getSize (int maxWidth) {
|
|
358 int textWidth = 0, messageWidth = 0;
|
|
359 int w, h;
|
|
360 if (layoutText !is null) {
|
|
361 OS.pango_layout_set_width (layoutText, -1);
|
|
362 OS.pango_layout_get_size (layoutText, &w, &h);
|
|
363 textWidth = OS.PANGO_PIXELS (w );
|
|
364 }
|
|
365 if (layoutMessage !is null) {
|
|
366 OS.pango_layout_set_width (layoutMessage, -1);
|
|
367 OS.pango_layout_get_size (layoutMessage, &w, &h);
|
|
368 messageWidth = OS.PANGO_PIXELS (w );
|
|
369 }
|
|
370 int messageTrim = 2 * INSET + 2 * BORDER + 2 * PADDING;
|
|
371 bool hasImage = layoutText !is null && (style & (SWT.ICON_ERROR | SWT.ICON_INFORMATION | SWT.ICON_WARNING)) !is 0;
|
|
372 int textTrim = messageTrim + (hasImage ? IMAGE_SIZE : 0);
|
|
373 int width = Math.min (maxWidth, Math.max (textWidth + textTrim, messageWidth + messageTrim));
|
|
374 int textHeight = 0, messageHeight = 0;
|
|
375 if (layoutText !is null) {
|
|
376 OS.pango_layout_set_width (layoutText, (maxWidth - textTrim) * OS.PANGO_SCALE);
|
|
377 OS.pango_layout_get_size (layoutText, &w, &h);
|
|
378 textHeight = OS.PANGO_PIXELS (h );
|
|
379 }
|
|
380 if (layoutMessage !is null) {
|
|
381 OS.pango_layout_set_width (layoutMessage, (maxWidth - messageTrim) * OS.PANGO_SCALE);
|
|
382 OS.pango_layout_get_size (layoutMessage, &w, &h);
|
|
383 messageHeight = OS.PANGO_PIXELS (h);
|
|
384 }
|
|
385 int height = 2 * BORDER + 2 * PADDING + messageHeight;
|
|
386 if (layoutText !is null) height += Math.max (IMAGE_SIZE, textHeight) + 2 * PADDING;
|
|
387 return new Point(width, height);
|
|
388 }
|
|
389
|
|
390 /**
|
|
391 * Returns the receiver's text, which will be an empty
|
|
392 * string if it has never been set.
|
|
393 *
|
|
394 * @return the receiver's text
|
|
395 *
|
|
396 * @exception SWTException <ul>
|
|
397 * <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
|
|
398 * <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
|
|
399 * </ul>
|
|
400 */
|
|
401 public char[] getText () {
|
|
402 checkWidget ();
|
|
403 return text;
|
|
404 }
|
|
405
|
|
406 /**
|
|
407 * Returns <code>true</code> if the receiver is visible, and
|
|
408 * <code>false</code> otherwise.
|
|
409 * <p>
|
|
410 * If one of the receiver's ancestors is not visible or some
|
|
411 * other condition makes the receiver not visible, this method
|
|
412 * may still indicate that it is considered visible even though
|
|
413 * it may not actually be showing.
|
|
414 * </p>
|
|
415 *
|
|
416 * @return the receiver's visibility state
|
|
417 *
|
|
418 * @exception SWTException <ul>
|
|
419 * <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
|
|
420 * <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
|
|
421 * </ul>
|
|
422 */
|
|
423 public bool getVisible () {
|
|
424 checkWidget ();
|
|
425 if ((style & SWT.BALLOON) !is 0) return OS.GTK_WIDGET_VISIBLE (handle);
|
|
426 auto tipWindow = OS.GTK_TOOLTIPS_TIP_WINDOW (cast(GtkTooltips*)handle);
|
|
427 return OS.GTK_WIDGET_VISIBLE (tipWindow);
|
|
428 }
|
|
429
|
|
430 override int /*long*/ gtk_button_press_event (GtkWidget* widget, GdkEventButton* event) {
|
|
431 notifyListeners (SWT.Selection, new Event ());
|
|
432 setVisible (false);
|
|
433 return 0;
|
|
434 }
|
|
435
|
|
436 override int /*long*/ gtk_expose_event (GtkWidget* widget, GdkEventExpose* event) {
|
|
437 auto window = OS.GTK_WIDGET_WINDOW (handle);
|
|
438 auto gdkGC = cast(GdkGC*)OS.gdk_gc_new (window);
|
|
439 OS.gdk_draw_polygon (window, gdkGC, 0, cast(GdkPoint*)borderPolygon.ptr, borderPolygon.length / 2);
|
|
440 int x = BORDER + PADDING;
|
|
441 int y = BORDER + PADDING;
|
|
442 if (spikeAbove) y += TIP_HEIGHT;
|
|
443 if (layoutText !is null) {
|
|
444 char[] buffer = null;
|
|
445 int id = style & (SWT.ICON_ERROR | SWT.ICON_INFORMATION | SWT.ICON_WARNING);
|
|
446 switch (id) {
|
|
447 case SWT.ICON_ERROR: buffer = "gtk-dialog-error"; break;
|
|
448 case SWT.ICON_INFORMATION: buffer = "gtk-dialog-info"; break;
|
|
449 case SWT.ICON_WARNING: buffer = "gtk-dialog-warning"; break;
|
|
450 }
|
|
451 if (buffer !is null) {
|
|
452 auto style = OS.gtk_widget_get_default_style ();
|
|
453 auto pixbuf = OS.gtk_icon_set_render_icon (
|
|
454 OS.gtk_icon_factory_lookup_default (buffer.ptr),
|
|
455 style,
|
|
456 OS.GTK_TEXT_DIR_NONE,
|
|
457 OS.GTK_STATE_NORMAL,
|
|
458 OS.GTK_ICON_SIZE_MENU,
|
|
459 null,
|
|
460 null);
|
|
461 OS.gdk_draw_pixbuf (window, gdkGC, pixbuf, 0, 0, x, y, IMAGE_SIZE, IMAGE_SIZE, OS.GDK_RGB_DITHER_NORMAL, 0, 0);
|
|
462 OS.g_object_unref (pixbuf);
|
|
463 x += IMAGE_SIZE;
|
|
464 }
|
|
465 x += INSET;
|
|
466 OS.gdk_draw_layout (window, gdkGC, x, y, layoutText);
|
|
467 int w, h;
|
|
468 OS.pango_layout_get_size (layoutText, &w, &h);
|
|
469 y += 2 * PADDING + Math.max (IMAGE_SIZE, OS.PANGO_PIXELS (h ));
|
|
470 }
|
|
471 if (layoutMessage !is null) {
|
|
472 x = BORDER + PADDING + INSET;
|
|
473 OS.gdk_draw_layout (window, gdkGC, x, y, layoutMessage);
|
|
474 }
|
|
475 OS.g_object_unref (gdkGC);
|
|
476 return 0;
|
|
477 }
|
|
478
|
|
479 override int /*long*/ gtk_size_allocate (GtkWidget* widget, int /*long*/ allocation) {
|
|
480 Point point = getLocation ();
|
|
481 int x = point.x;
|
|
482 int y = point.y;
|
|
483 auto screen = OS.gdk_screen_get_default ();
|
|
484 OS.gtk_widget_realize (widget);
|
|
485 int monitorNumber = OS.gdk_screen_get_monitor_at_window (screen, OS.GTK_WIDGET_WINDOW (widget));
|
|
486 GdkRectangle* dest = new GdkRectangle ();
|
|
487 OS.gdk_screen_get_monitor_geometry (screen, monitorNumber, dest);
|
|
488 int w = OS.GTK_WIDGET_WIDTH (widget);
|
|
489 int h = OS.GTK_WIDGET_HEIGHT (widget);
|
|
490 if (dest.height < y + h) y -= h;
|
|
491 if (dest.width < x + w) x -= w;
|
|
492 OS.gtk_window_move (cast(GtkWindow*)widget, x, y);
|
|
493 return 0;
|
|
494 }
|
|
495
|
|
496 void hookEvents () {
|
|
497 if ((style & SWT.BALLOON) !is 0) {
|
|
498 OS.g_signal_connect_closure (handle, OS.expose_event.ptr, display.closures [EXPOSE_EVENT], false);
|
|
499 OS.gtk_widget_add_events (handle, OS.GDK_BUTTON_PRESS_MASK);
|
|
500 OS.g_signal_connect_closure (handle, OS.button_press_event.ptr, display.closures [BUTTON_PRESS_EVENT], false);
|
|
501 } else {
|
|
502 auto tipWindow = OS.GTK_TOOLTIPS_TIP_WINDOW (cast(GtkTooltips*)handle);
|
|
503 if (tipWindow !is null) {
|
|
504 OS.g_signal_connect_closure (tipWindow, OS.size_allocate.ptr, display.closures [SIZE_ALLOCATE], false);
|
|
505 OS.gtk_widget_add_events (tipWindow, OS.GDK_BUTTON_PRESS_MASK);
|
|
506 OS.g_signal_connect_closure (tipWindow, OS.button_press_event.ptr, display.closures [BUTTON_PRESS_EVENT], false);
|
|
507 }
|
|
508 }
|
|
509 }
|
|
510
|
|
511 /**
|
|
512 * Returns <code>true</code> if the receiver is visible and all
|
|
513 * of the receiver's ancestors are visible and <code>false</code>
|
|
514 * otherwise.
|
|
515 *
|
|
516 * @return the receiver's visibility state
|
|
517 *
|
|
518 * @exception SWTException <ul>
|
|
519 * <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
|
|
520 * <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
|
|
521 * </ul>
|
|
522 *
|
|
523 * @see #getVisible
|
|
524 */
|
|
525 public bool isVisible () {
|
|
526 checkWidget ();
|
|
527 return getVisible ();
|
|
528 }
|
|
529
|
|
530 void register () {
|
|
531 super.register ();
|
|
532 if ((style & SWT.BALLOON) is 0) {
|
|
533 auto tipWindow = OS.GTK_TOOLTIPS_TIP_WINDOW (cast(GtkTooltips*)handle);
|
|
534 if (tipWindow !is null) display.addWidget (tipWindow, this);
|
|
535 }
|
|
536 }
|
|
537
|
|
538 void releaseWidget () {
|
|
539 super.releaseWidget ();
|
|
540 if (layoutText !is null) OS.g_object_unref (layoutText);
|
|
541 layoutText = null;
|
|
542 if (layoutMessage !is null) OS.g_object_unref (layoutMessage);
|
|
543 layoutMessage = null;
|
|
544 if (timerId !is 0) OS.gtk_timeout_remove(timerId);
|
|
545 timerId = 0;
|
|
546 text = null;
|
|
547 message = null;
|
|
548 borderPolygon = null;
|
|
549 }
|
|
550
|
|
551 /**
|
|
552 * Removes the listener from the collection of listeners who will
|
|
553 * be notified when the receiver is selected by the user.
|
|
554 *
|
|
555 * @param listener the listener which should no longer be notified
|
|
556 *
|
|
557 * @exception IllegalArgumentException <ul>
|
|
558 * <li>ERROR_NULL_ARGUMENT - if the listener is null</li>
|
|
559 * </ul>
|
|
560 * @exception SWTException <ul>
|
|
561 * <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
|
|
562 * <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
|
|
563 * </ul>
|
|
564 *
|
|
565 * @see SelectionListener
|
|
566 * @see #addSelectionListener
|
|
567 */
|
|
568 public void removeSelectionListener (SelectionListener listener) {
|
|
569 checkWidget();
|
|
570 if (listener is null) error (SWT.ERROR_NULL_ARGUMENT);
|
|
571 if (eventTable is null) return;
|
|
572 eventTable.unhook (SWT.Selection, listener);
|
|
573 eventTable.unhook (SWT.DefaultSelection, listener);
|
|
574 }
|
|
575
|
|
576 /**
|
|
577 * Makes the receiver hide automatically when <code>true</code>,
|
|
578 * and remain visible when <code>false</code>.
|
|
579 *
|
|
580 * @param autoHide the auto hide state
|
|
581 *
|
|
582 * @exception SWTException <ul>
|
|
583 * <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
|
|
584 * <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
|
|
585 * </ul>
|
|
586 *
|
|
587 * @see #getVisible
|
|
588 * @see #setVisible
|
|
589 */
|
|
590 public void setAutoHide (bool autohide) {
|
|
591 checkWidget ();
|
|
592 this.autohide = autohide;
|
|
593 //TODO - update when visible
|
|
594 }
|
|
595
|
|
596 /**
|
|
597 * Sets the location of the receiver, which must be a tooltip,
|
|
598 * to the point specified by the arguments which are relative
|
|
599 * to the display.
|
|
600 * <p>
|
|
601 * Note that this is different from most widgets where the
|
|
602 * location of the widget is relative to the parent.
|
|
603 * </p>
|
|
604 *
|
|
605 * @param x the new x coordinate for the receiver
|
|
606 * @param y the new y coordinate for the receiver
|
|
607 *
|
|
608 * @exception SWTException <ul>
|
|
609 * <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
|
|
610 * <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
|
|
611 * </ul>
|
|
612 */
|
|
613 public void setLocation (int x, int y) {
|
|
614 checkWidget ();
|
|
615 this.x = x;
|
|
616 this.y = y;
|
|
617 if ((style & SWT.BALLOON) !is 0) {
|
|
618 if (OS.GTK_WIDGET_VISIBLE (handle)) configure ();
|
|
619 } else {
|
|
620 auto tipWindow = OS.GTK_TOOLTIPS_TIP_WINDOW (cast(GtkTooltips*)handle);
|
|
621 if (OS.GTK_WIDGET_VISIBLE (tipWindow)) {
|
|
622 OS.gtk_window_move (cast(GtkWindow*)tipWindow, x, y);
|
|
623 }
|
|
624 }
|
|
625 }
|
|
626
|
|
627 /**
|
|
628 * Sets the location of the receiver, which must be a tooltip,
|
|
629 * to the point specified by the argument which is relative
|
|
630 * to the display.
|
|
631 * <p>
|
|
632 * Note that this is different from most widgets where the
|
|
633 * location of the widget is relative to the parent.
|
|
634 * </p><p>
|
|
635 * Note that the platform window manager ultimately has control
|
|
636 * over the location of tooltips.
|
|
637 * </p>
|
|
638 *
|
|
639 * @param location the new location for the receiver
|
|
640 *
|
|
641 * @exception IllegalArgumentException <ul>
|
|
642 * <li>ERROR_NULL_ARGUMENT - if the point is null</li>
|
|
643 * </ul>
|
|
644 * @exception SWTException <ul>
|
|
645 * <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
|
|
646 * <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
|
|
647 * </ul>
|
|
648 */
|
|
649 public void setLocation (Point location) {
|
|
650 checkWidget ();
|
|
651 if (location is null) SWT.error (SWT.ERROR_NULL_ARGUMENT);
|
|
652 setLocation (location.x, location.y);
|
|
653 }
|
|
654
|
|
655 /**
|
|
656 * Sets the receiver's message.
|
|
657 *
|
|
658 * @param string the new message
|
|
659 *
|
|
660 * @exception IllegalArgumentException <ul>
|
|
661 * <li>ERROR_NULL_ARGUMENT - if the text is null</li>
|
|
662 * </ul>
|
|
663 * @exception SWTException <ul>
|
|
664 * <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
|
|
665 * <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
|
|
666 * </ul>
|
|
667 */
|
|
668 public void setMessage (char[] string) {
|
|
669 checkWidget ();
|
|
670 if (string is null) error (SWT.ERROR_NULL_ARGUMENT);
|
|
671 message = string;
|
|
672 if ((style & SWT.BALLOON) is 0) return;
|
|
673 if (layoutMessage !is null) OS.g_object_unref (layoutMessage);
|
|
674 layoutMessage = null;
|
|
675 if (message.length !is 0) {
|
|
676 layoutMessage = OS.gtk_widget_create_pango_layout (handle, toStringz( message ));
|
|
677 OS.pango_layout_set_wrap (layoutMessage, OS.PANGO_WRAP_WORD_CHAR);
|
|
678 }
|
|
679 if (OS.GTK_WIDGET_VISIBLE (handle)) configure ();
|
|
680 }
|
|
681
|
|
682 /**
|
|
683 * Sets the receiver's text.
|
|
684 *
|
|
685 * @param string the new text
|
|
686 *
|
|
687 * @exception IllegalArgumentException <ul>
|
|
688 * <li>ERROR_NULL_ARGUMENT - if the text is null</li>
|
|
689 * </ul>
|
|
690 * @exception SWTException <ul>
|
|
691 * <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
|
|
692 * <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
|
|
693 * </ul>
|
|
694 */
|
|
695 public void setText (char[] string) {
|
|
696 checkWidget ();
|
|
697 if (string is null) error (SWT.ERROR_NULL_ARGUMENT);
|
|
698 text = string;
|
|
699 if ((style & SWT.BALLOON) is 0) return;
|
|
700 if (layoutText !is null) OS.g_object_unref (layoutText);
|
|
701 layoutText = null;
|
|
702 if (text.length !is 0) {
|
|
703 layoutText = OS.gtk_widget_create_pango_layout (handle, toStringz(text));
|
|
704 auto boldAttr = OS.pango_attr_weight_new (OS.PANGO_WEIGHT_BOLD);
|
|
705 boldAttr.start_index = 0;
|
|
706 boldAttr.end_index = text.length+1;
|
|
707 auto attrList = OS.pango_attr_list_new ();
|
|
708 OS.pango_attr_list_insert (attrList, boldAttr);
|
|
709 OS.pango_layout_set_attributes (layoutText, attrList);
|
|
710 OS.pango_attr_list_unref (attrList);
|
|
711 OS.pango_layout_set_wrap (layoutText, OS.PANGO_WRAP_WORD_CHAR);
|
|
712 }
|
|
713 if (OS.GTK_WIDGET_VISIBLE (handle)) configure ();
|
|
714 }
|
|
715
|
|
716 /**
|
|
717 * Marks the receiver as visible if the argument is <code>true</code>,
|
|
718 * and marks it invisible otherwise.
|
|
719 * <p>
|
|
720 * If one of the receiver's ancestors is not visible or some
|
|
721 * other condition makes the receiver not visible, marking
|
|
722 * it visible may not actually cause it to be displayed.
|
|
723 * </p>
|
|
724 *
|
|
725 * @param visible the new visibility state
|
|
726 *
|
|
727 * @exception SWTException <ul>
|
|
728 * <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
|
|
729 * <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
|
|
730 * </ul>
|
|
731 */
|
|
732 public void setVisible (bool visible) {
|
|
733 if (timerId !is 0) OS.gtk_timeout_remove(timerId);
|
|
734 timerId = 0;
|
|
735 if (visible) {
|
|
736 if ((style & SWT.BALLOON) !is 0) {
|
|
737 configure ();
|
|
738 OS.gtk_widget_show (handle);
|
|
739 } else {
|
|
740 auto vboxHandle = parent.vboxHandle;
|
|
741 char[] string = text;
|
|
742 if (text.length > 0) string ~= "\n\n";
|
|
743 string ~= message;
|
|
744 char* buffer = toStringz( string );
|
|
745 OS.gtk_tooltips_set_tip (cast(GtkTooltips*)handle, vboxHandle, buffer, null);
|
|
746 auto data = OS.gtk_tooltips_data_get (vboxHandle);
|
|
747 OS.GTK_TOOLTIPS_SET_ACTIVE (cast(GtkTooltips*)handle, data);
|
|
748 OS.gtk_tooltips_set_tip (cast(GtkTooltips*)handle, vboxHandle, buffer, null);
|
|
749 }
|
|
750 if (autohide) timerId = display.doWindowTimerAdd( &timerProcCallbackData, DELAY, handle);
|
|
751 } else {
|
|
752 if ((style & SWT.BALLOON) !is 0) {
|
|
753 OS.gtk_widget_hide (handle);
|
|
754 } else {
|
|
755 auto tipWindow = OS.GTK_TOOLTIPS_TIP_WINDOW (cast(GtkTooltips*)handle);
|
|
756 OS.gtk_widget_hide (tipWindow);
|
|
757 }
|
|
758 }
|
|
759 }
|
|
760
|
|
761 override int /*long*/ timerProc (GtkWidget* widget) {
|
|
762 if ((style & SWT.BALLOON) !is 0) {
|
|
763 OS.gtk_widget_hide (handle);
|
|
764 } else {
|
|
765 auto tipWindow = OS.GTK_TOOLTIPS_TIP_WINDOW (cast(GtkTooltips*)handle);
|
|
766 OS.gtk_widget_hide (tipWindow);
|
|
767 }
|
|
768 return 0;
|
|
769 }
|
|
770
|
|
771 }
|