-
Notifications
You must be signed in to change notification settings - Fork 0
/
SystemTrayController.java
418 lines (350 loc) · 14.2 KB
/
SystemTrayController.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
package com.bookiescrape.app.sample;
import java.awt.AWTException;
import java.awt.Desktop;
import java.awt.Image;
import java.awt.MenuItem;
import java.awt.PopupMenu;
import java.awt.SystemTray;
import java.awt.Taskbar;
import java.awt.Toolkit;
import java.awt.TrayIcon;
import java.awt.TrayIcon.MessageType;
import java.awt.desktop.AppForegroundEvent;
import java.awt.desktop.AppForegroundListener;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.io.IOException;
import java.net.URI;
import java.net.URL;
import javafx.application.Platform;
/**
* Handles the creation and layout of the system tray icon and system tray
* application popup menu.
*
* @author Jonathan Henly
* @see SystemTray
*/
public class SystemTrayController {
private SystemTray tray;
private TrayIcon trayIcon;
private Image defIcon;
private URI aboutUri;
private MenuItem about;
private MenuItem runScraper;
private MenuItem dashboard;
private MenuItem settings;
private MenuItem logs;
private MenuItem quit;
/**************************************************************************
* *
* Constructor(s) *
* *
*************************************************************************/
/**
* Creates a system tray controller with the application's system tray icon
* image set to the default system tray icon image.
*
* @throws UnsupportedOperationException if the system tray is not supported
*/
public SystemTrayController() throws UnsupportedOperationException {
this(null);
}
/**
* Creates a system tray controller and sets the application's system tray
* icon image to the image loaded from the specified image url.
*
* @param iconImageUrl - the url to load the tray icon's image from, or
* {@code null} if the default system tray icon image should be used
* @throws UnsupportedOperationException if the system tray is not supported
*/
public SystemTrayController(URL iconImageUrl) throws UnsupportedOperationException {
// ensure awt toolkit is initialized.
java.awt.Toolkit.getDefaultToolkit();
// assign system tray or throw if not supported
tray = getSystemTrayOrThrowIfNotSupported();
// listen for app raised to foreground/moved to background events
Desktop.getDesktop().addAppEventListener(new AppFocusListener());
// load specified icon image or use the default toolkit icon image
defIcon = loadIconImageOrUseDefault(iconImageUrl);
// create tray icon and add it to the system tray
initTrayIcon();
addAppTrayIconToSystemTray();
}
/**************************************************************************
* *
* Public API *
* *
*************************************************************************/
/**
* Displays a system tray notification to the user.
* @param caption - the notification caption
* @param text - the notification detail text
* @param type - the type of notification
* @see MessageType
*/
public void notifyUser(String caption, String text, MessageType type) {
runLater(() -> trayIcon.displayMessage(caption, text, type));
}
/**
* Shuts down the system tray if it's supported, otherwise calling this
* method is a no-op.
*/
public void shutdown() {
if (trayIcon == null) { return; }
System.out.println("SystemTrayController: closing");
runLater(() -> {
Platform.exit();
tray.remove(trayIcon);
});
}
/**
* Specifies the system tray icon's tool-tip message.
* @param tip - the icon's tool-tip message to display
*/
public void setToolTip(String tip) {
if (trayIcon == null) { return; }
runLater(() -> trayIcon.setToolTip(tip));
}
/**
* Sets a status string at the first position in the context menu.
* <p>
* This status string appears as a disabled menu entry.
* @param status - the string to display at the first position in the
* context menu
* @see SystemTray#setStatus(String)
*/
public void setStatus(String status) {
if (trayIcon == null) { return; }
// javax.swing.SwingUtilities.invokeLater(() -> trayIcon.get().);
}
/**
* Specifies the image to use for the system tray icon.
* @param imageUrl - image to use for the system tray icon
* @see SystemTray#setImage(URL)
*/
public void setIconImage(URL imageUrl) {
if (trayIcon == null) { return; }
runLater(() -> {
defIcon = loadIconImageOrUseDefault(imageUrl);
trayIcon.setImage(defIcon);
});
}
/**
* Specifies the URL of the website to open in a browser when the 'About'
* menu item is selected.
* @param url - the website URL to open when the 'About' menu item is
* selected
*/
public void setAboutBrowserUrl(String url) {
runLater(() -> {
try {
aboutUri = URI.create(url);
} catch (Exception e) {
// TODO log create URI exception rather than printing stack trace
e.printStackTrace();
}
});
}
/**
* Specifies the enabled state of the 'Run Scraper' menu item.
* @param enabled - the enabled state of the 'Run Scraper' menu item
*/
public void setRunScraperEnabled(boolean enabled) {
runLater(() -> runScraper.setEnabled(enabled));
}
/**
* Sets the about tray icon menu item selected action listener.
* @param listener - the about selected listener
*/
public void setOnAboutSelected(ActionListener listener) {
setMenuItemActionListener(about, listener);
}
/**
* Sets the run scraper tray icon menu item selected action listener.
* @param listener - the run scraper selected listener
*/
public void setOnRunScraperSelected(ActionListener listener) {
setMenuItemActionListener(runScraper, listener);
}
/**
* Sets the dashboard tray icon menu item selected action listener.
* @param listener - the dashboard selected listener
*/
public void setOnDashboardSelected(ActionListener listener) {
setMenuItemActionListener(dashboard, listener);
}
/**
* Sets the settings tray icon menu item selected action listener.
* @param listener - the settings selected listener
*/
public void setOnSettingsSelected(ActionListener listener) {
setMenuItemActionListener(settings, listener);
}
/**
* Sets the view logs tray icon menu item selected action listener.
* @param listener - the view logs selected listener
*/
public void setOnViewLogsSelected(ActionListener listener) {
setMenuItemActionListener(logs, listener);
}
/**
* Sets the quit tray icon menu item selected action listener.
* <p>
* The quit tray icon menu item has a preset action listener that handles
* exiting of the application, therefore changing the action listener
* should rarely, if ever, be needed.
* @param listener - the quit selected listener
*/
public void setOnQuitSelected(ActionListener listener) {
setMenuItemActionListener(quit, listener);
}
/**************************************************************************
* *
* Private API *
* *
*************************************************************************/
/** Gets the system tray or throws an UnsupportedOperationException if the
* system tray is not supported.
*/
private SystemTray getSystemTrayOrThrowIfNotSupported() {
UnsupportedOperationException uoe = new UnsupportedOperationException("system tray not supported.");
if (!SystemTray.isSupported()) { throw uoe; }
SystemTray tmp = null;
try {
tmp = SystemTray.getSystemTray();
} catch (Exception e) {
throw uoe;
}
return tmp;
}
/** Initializes the tray icon. */
private void initTrayIcon() {
trayIcon = new TrayIcon(defIcon, Main.APP_NAME);
// allow the system to auto-size the tray icon
trayIcon.setImageAutoSize(true);
// set the tray icon's tool-tip
// trayIcon.get().setToolTip("");
// add notification click listener to tray icon
trayIcon.setActionCommand("notification-clicked");
trayIcon.addActionListener(this::onNotificationClicked);
// create and set tray icon's popup menu
trayIcon.setPopupMenu(createPopupMenu());
}
/**
* Helper that loads the specified icon image url or uses the default
* toolkit icon image.
*/
private Image loadIconImageOrUseDefault(URL iconImageUrl) {
Image icon = null;
if (iconImageUrl != null) {
try {
icon = Toolkit.getDefaultToolkit().createImage(iconImageUrl);
} catch (Exception e) {
// TODO log if an exception occurred while loading the icon image
icon = Taskbar.getTaskbar().getIconImage();
}
} else {
icon = Taskbar.getTaskbar().getIconImage();
}
return icon;
}
/** Adds application tray icon to system tray. */
private void addAppTrayIconToSystemTray() {
try {
tray.add(trayIcon);
} catch (AWTException e) {
// TODO log tray add exception rather than printing stack trace
e.printStackTrace();
}
}
/**
* Handles clicking actions on notifications.
* @param action - the clicking action to handle
*/
private void onNotificationClicked(ActionEvent action) {
System.out.println(action.getActionCommand());
System.out.println("Notification Clicked");
}
/** Listeners for app raised to foreground/moved to background events. */
private class AppFocusListener implements AppForegroundListener {
@Override
public void appRaisedToForeground(AppForegroundEvent e) {
System.out.println("SystemTrayController: app moved to foreground");
}
@Override
public void appMovedToBackground(AppForegroundEvent e) {
System.out.println("SystemTrayController: app moved to background");
}
} // class AppFocusListener
/** Convenience helper that invokes {@code SwingUtils.invokeLater(toRun)}. */
private void runLater(final Runnable toRun) {
javax.swing.SwingUtilities.invokeLater(toRun);
}
/** Helper method used by setOn*Selected methods. */
private void setMenuItemActionListener(MenuItem menuItem, ActionListener listener) {
runLater(() -> {
// remove any action listeners from menu item
for (ActionListener al : menuItem.getActionListeners()) { menuItem.removeActionListener(al); }
// add specified action listener to menu item
menuItem.addActionListener(listener);
});
}
/**************************************************************************
* *
* Popup Menu Creation *
* *
*************************************************************************/
/** Creates the menu items and lays out the tray icon's popup menu. */
private PopupMenu createPopupMenu() {
// init the menu items before adding them to the menu
initMenuItems();
PopupMenu menu = new PopupMenu();
menu.add(about);
menu.add(runScraper);
menu.addSeparator();
menu.add(dashboard);
menu.add(settings);
menu.add(logs);
menu.addSeparator();
menu.add(quit);
return menu;
}
/** Initializes the popup menu's items. */
private void initMenuItems() {
runScraper = new MenuItem("Run Scraper");
dashboard = new MenuItem("Dashboard");
settings = new MenuItem("Settings");
logs = new MenuItem("View Logs");
// about and quit have default action listeners
initAboutMenuItem();
initQuitMenuItem();
}
/**
* Creates about menu item and sets its action listener to open a browser
* to the application's website.
*/
private void initAboutMenuItem() {
about = new MenuItem("About");
about.addActionListener(e -> {
System.out.println("About was selected");
try {
Desktop.getDesktop().browse(aboutUri);
} catch (IOException e1) {
// TODO log browser exception rather than print stack trace
e1.printStackTrace();
}
});
}
/**
* Creates quit menu item and sets its action listener to handle exiting of
* the application and removal of the system tray icon.
*/
private void initQuitMenuItem() {
quit = new MenuItem("Quit Bookie Scrape");
quit.addActionListener(e -> {
System.out.println("Quit was selected");
Platform.exit();
tray.remove(trayIcon);
});
}
} // class SystemTrayController