diff options
author | Vincent Lucas <chenzo@jitsi.org> | 2012-10-17 17:03:33 +0000 |
---|---|---|
committer | Vincent Lucas <chenzo@jitsi.org> | 2012-10-17 17:03:33 +0000 |
commit | 81c971944b8a28235abc9e277ada7e306388ba53 (patch) | |
tree | b92897317460b2ddee1c0f3956b5b56f6a41c6af | |
parent | a754cc94e0a1a37bb2b590b31d72b1637ec2d9da (diff) | |
download | jitsi-81c971944b8a28235abc9e277ada7e306388ba53.zip jitsi-81c971944b8a28235abc9e277ada7e306388ba53.tar.gz jitsi-81c971944b8a28235abc9e277ada7e306388ba53.tar.bz2 |
Adds a notification popup when the audio device configuration has changed.
7 files changed, 252 insertions, 130 deletions
diff --git a/resources/languages/resources.properties b/resources/languages/resources.properties index 25fa783..f3e8066 100644 --- a/resources/languages/resources.properties +++ b/resources/languages/resources.properties @@ -1300,6 +1300,7 @@ plugin.notificationconfig.event.DTMFTone.8=DTMF tone 8 plugin.notificationconfig.event.DTMFTone.9=DTMF tone 9 plugin.notificationconfig.event.DTMFTone.*=DTMF tone * plugin.notificationconfig.event.DTMFTone.#=DTMF tone # +plugin.notificationconfig.event.DeviceConfigurationChanged=Device configuration change # ZRTP Securing impl.media.security.WARNING_NO_RS_MATCH=<html>No retained shared secret available.<br/><b>SAS verification is recommended</b></html> @@ -1356,8 +1357,8 @@ impl.media.configform.VIDEO_RESOLUTION=Video resolution impl.media.configform.VIDEO_FRAME_RATE=Custom frame rate (per sec.) impl.media.configform.VIDEO_PACKETS_POLICY=Maximum allowed bandwidth (kBytes/s) impl.media.configform.VIDEO_RESET=Reset defaults -impl.media.configform.AUDIO_DEVICE_CONFIG=Audio device configuration -impl.media.configform.AUDIO_DEVICE_CONNECTED_REMOVED=You have connected or removed an audio device from your computer. How would you like to use the new configuration? +impl.media.configform.AUDIO_DEVICE_CONFIG_CHANGED=Audio configuration has changed +impl.media.configform.AUDIO_DEVICE_CONFIG_MANAGMENT_CLICK=Clic here for device management impl.neomedia.configform.AUDIO=Audio impl.neomedia.configform.VIDEO=Video diff --git a/src/net/java/sip/communicator/impl/neomedia/NeomediaActivator.java b/src/net/java/sip/communicator/impl/neomedia/NeomediaActivator.java index c2acd14..edf966b 100644 --- a/src/net/java/sip/communicator/impl/neomedia/NeomediaActivator.java +++ b/src/net/java/sip/communicator/impl/neomedia/NeomediaActivator.java @@ -15,7 +15,10 @@ import javax.swing.*; import net.java.sip.communicator.impl.neomedia.codec.video.h264.*; import net.java.sip.communicator.service.gui.*; +import net.java.sip.communicator.service.notification.*; import net.java.sip.communicator.service.resources.*; +import net.java.sip.communicator.service.systray.*; +import net.java.sip.communicator.service.systray.event.*; import net.java.sip.communicator.util.*; import net.java.sip.communicator.util.swing.*; @@ -86,6 +89,13 @@ public class NeomediaActivator = "net.java.sip.communicator.impl.neomedia.callrecordingconfig.DISABLED"; /** + * The name of the notification pop-up event displayed when the device + * configration has changed. + */ + private static final String DEVICE_CONFIGURATION_HAS_CHANGED + = "DeviceConfigurationChanged"; + + /** * The context in which the one and only <tt>NeomediaActivator</tt> instance * has started executing. */ @@ -105,6 +115,11 @@ public class NeomediaActivator private static FileAccessService fileAccessService; /** + * The notifcation service to pop-up messages. + */ + private static NotificationService notificationService; + + /** * The one and only <tt>MediaServiceImpl</tt> instance registered in * {@link #bundleContext} by the <tt>NeomediaActivator</tt> instance. */ @@ -124,7 +139,12 @@ public class NeomediaActivator */ private static PacketLoggingService packetLoggingService = null; - private PropertyChangeListener deviceConfigurationPropertyChangeListener; + /** + * A listener to the click on the popup message concerning device + * configuration changes. + */ + private AudioDeviceConfigurationListener + deviceConfigurationPropertyChangeListener; /** * Audio configuration dialog. @@ -137,6 +157,12 @@ public class NeomediaActivator private static MediaConfigurationImpl mediaConfiguration; /** + * The audio configuration form used to define the capture/notify/playback + * audio devices. + */ + private static ConfigurationForm audioConfigurationForm; + + /** * Starts the execution of the neomedia bundle in the specified context. * * @param bundleContext the context in which the neomedia bundle is to start @@ -179,7 +205,7 @@ public class NeomediaActivator // If the audio configuration form is disabled don't register it. if ((cfg == null) || !cfg.getBoolean(AUDIO_CONFIG_DISABLED_PROP, false)) { - final ConfigurationForm audioConfigurationForm + audioConfigurationForm = new LazyConfigurationForm( AudioConfigurationPanel.class.getName(), getClass().getClassLoader(), @@ -194,18 +220,12 @@ public class NeomediaActivator if (deviceConfigurationPropertyChangeListener == null) { + // Initializes and registers the changed device configuration + // event ot the notification service. + getNotificationService(); + deviceConfigurationPropertyChangeListener - = new PropertyChangeListener() - { - public void propertyChange(PropertyChangeEvent event) - { - if (DeviceConfiguration.PROP_AUDIO_SYSTEM_DEVICES - .equals(event.getPropertyName())) - { - showAudioConfiguration(); - } - } - }; + = new AudioDeviceConfigurationListener(); mediaServiceImpl .getDeviceConfiguration() .addPropertyChangeListener( @@ -317,118 +337,6 @@ public class NeomediaActivator } /** - * Show audio configuration panel when media devices change. - */ - private void showAudioConfiguration() - { - if(audioConfigDialog != null && audioConfigDialog.isVisible()) - { - // Because toFront() method gives us no guarantee that our dialog - // would go on top we'll try to also first request the focus and - // set our dialog always on top to put all the chances on our side. - audioConfigDialog.requestFocus(); - audioConfigDialog.setAlwaysOnTop(true); - audioConfigDialog.toFront(); - audioConfigDialog.setAlwaysOnTop(false); - return; - } - - if (!SwingUtilities.isEventDispatchThread()) - { - SwingUtilities.invokeLater( - new Runnable() - { - public void run() - { - showAudioConfiguration(); - } - }); - return; - } - - audioConfigDialog = - new SIPCommDialog() - { - /** Serial version UID. */ - private static final long serialVersionUID = 0L; - - /** {@inheritDoc} */ - @Override - protected void close(boolean escaped) - { - setVisible(false); - audioConfigDialog = null; - } - }; - - TransparentPanel mainPanel - = new TransparentPanel(new BorderLayout(20, 5)); - TransparentPanel fieldsPanel - = new TransparentPanel(new BorderLayout(10, 5)); - - mainPanel.setBorder( - BorderFactory.createEmptyBorder(20, 20, 20, 20)); - - TransparentPanel btnPanel - = new TransparentPanel(new FlowLayout(FlowLayout.RIGHT)); - ResourceManagementService resources - = NeomediaActivator.getResources(); - JButton btn - = new JButton(resources.getI18NString("service.gui.CLOSE")); - - btn.addActionListener( - new ActionListener() - { - public void actionPerformed(ActionEvent evt) - { - audioConfigDialog.setVisible(false); - } - }); - btnPanel.add(btn); - - JTextArea infoTextArea = new JTextArea(); - - infoTextArea.setOpaque(false); - infoTextArea.setEditable(false); - infoTextArea.setWrapStyleWord(true); - infoTextArea.setLineWrap(true); - infoTextArea.setText( - resources.getI18NString( - "impl.media.configform" - + ".AUDIO_DEVICE_CONNECTED_REMOVED")); - - JPanel preview = new TransparentPanel(new GridBagLayout()); - mediaConfiguration.createAudioSystemControls( - mediaServiceImpl.getDeviceConfiguration().getAudioSystem(), - preview); - - fieldsPanel.add(infoTextArea, BorderLayout.NORTH); - fieldsPanel.add(preview, BorderLayout.CENTER); - fieldsPanel.add(btnPanel, BorderLayout.SOUTH); - - TransparentPanel iconPanel - = new TransparentPanel(new BorderLayout()); - - iconPanel.add( - new JLabel( - resources.getImage( - "plugin.mediaconfig.AUDIO_ICON_64x64")), - BorderLayout.NORTH); - - mainPanel.add(iconPanel, BorderLayout.WEST); - mainPanel.add(fieldsPanel, BorderLayout.CENTER); - - audioConfigDialog.setTitle( - resources.getI18NString( - "impl.media.configform.AUDIO_DEVICE_CONFIG")); - audioConfigDialog.add(mainPanel); - audioConfigDialog.validate(); - audioConfigDialog.pack(); - - audioConfigDialog.setVisible(true); - } - - /** * Stops the execution of the neomedia bundle in the specified context. * * @param bundleContext the context in which the neomedia bundle is to stop @@ -447,7 +355,12 @@ public class NeomediaActivator .getDeviceConfiguration() .removePropertyChangeListener( deviceConfigurationPropertyChangeListener); - deviceConfigurationPropertyChangeListener = null; + if(deviceConfigurationPropertyChangeListener != null) + { + deviceConfigurationPropertyChangeListener + .managePopupMessageListenerRegistration(false); + deviceConfigurationPropertyChangeListener = null; + } } } finally @@ -554,4 +467,157 @@ public class NeomediaActivator } return packetLoggingService; } + + /** + * Returns the <tt>NotificationService</tt> obtained from the bundle + * context. + * + * @return The <tt>NotificationService</tt> obtained from the bundle + * context. + */ + public static NotificationService getNotificationService() + { + if(notificationService == null) + { + // Get the notification service implementation + ServiceReference notifReference = bundleContext + .getServiceReference(NotificationService.class.getName()); + + notificationService = (NotificationService) bundleContext + .getService(notifReference); + + if(notificationService != null) + { + // Register a popup message for a device configuration changed + // notification. + notificationService.registerDefaultNotificationForEvent( + DEVICE_CONFIGURATION_HAS_CHANGED, + net.java.sip.communicator.service.notification.NotificationAction.ACTION_POPUP_MESSAGE, + "Device onfiguration has changed", + null); + } + } + + return notificationService; + } + + /** + * A listener to the click on the popup message concerning device + * configuration changes. + */ + private class AudioDeviceConfigurationListener + implements PropertyChangeListener, + SystrayPopupMessageListener + { + /** + * A boolean used to verify that this listener registers only once to + * the popup message notification handler. + */ + private boolean isRegisteredToPopupMessageListener = false; + + /** + * Registers or unregister as a popup message listener to detect when a + * user click on notification saying that the device configuration has + * changed. + * + * @param enable True to register to the popup message notifcation + * handler. False to unregister. + */ + public void managePopupMessageListenerRegistration(boolean enable) + { + Iterator<NotificationHandler> notificationHandlers + = notificationService.getActionHandlers( + net.java.sip.communicator.service.notification.NotificationAction.ACTION_POPUP_MESSAGE) + .iterator(); + NotificationHandler notificationHandler; + while(notificationHandlers.hasNext()) + { + notificationHandler = notificationHandlers.next(); + if(notificationHandler + instanceof PopupMessageNotificationHandler) + { + // Register. + if(enable) + { + ((PopupMessageNotificationHandler) notificationHandler) + .addPopupMessageListener(this); + } + // Unregister. + else + { + ((PopupMessageNotificationHandler) notificationHandler) + .removePopupMessageListener(this); + } + } + } + } + + /** + * Fonction called when an audio device is plugged or unplugged. + * + * @param The property change event which may concern the audio device. + */ + public void propertyChange(PropertyChangeEvent event) + { + if (DeviceConfiguration.PROP_AUDIO_SYSTEM_DEVICES + .equals(event.getPropertyName())) + { + NotificationService notificationService + = getNotificationService(); + if(notificationService != null) + { + // Registers only once to the popup message notification + // handler. + if(!isRegisteredToPopupMessageListener) + { + isRegisteredToPopupMessageListener = true; + managePopupMessageListenerRegistration(true); + } + // Fires the popup notification. + ResourceManagementService resources + = NeomediaActivator.getResources(); + notificationService.fireNotification( + DEVICE_CONFIGURATION_HAS_CHANGED, + resources.getI18NString( + "impl.media.configform" + + ".AUDIO_DEVICE_CONFIG_CHANGED"), + resources.getI18NString( + "impl.media.configform" + + ".AUDIO_DEVICE_CONFIG_MANAGMENT_CLICK"), + null, + this); + } + } + } + + /** + * Indicates that user has clicked on the systray popup message. + * + * @param evt the event triggered when user clicks on the systray popup + * message + */ + public void popupMessageClicked(SystrayPopupMessageEvent evt) + { + // Checks if this event is fired from one click on one of our popup + // message. + if(evt.getTag() == deviceConfigurationPropertyChangeListener) + { + // Get the UI service + ServiceReference uiReference = bundleContext + .getServiceReference(UIService.class.getName()); + + UIService uiService = (UIService) bundleContext + .getService(uiReference); + + if(uiService != null) + { + // Shows the audio configuration window. + ConfigurationContainer configurationContainer + = uiService.getConfigurationContainer(); + configurationContainer.setSelected(audioConfigurationForm); + configurationContainer.setVisible(true); + } + } + } + } } diff --git a/src/net/java/sip/communicator/impl/neomedia/neomedia.manifest.mf b/src/net/java/sip/communicator/impl/neomedia/neomedia.manifest.mf index 6aaf396..510e37f 100644 --- a/src/net/java/sip/communicator/impl/neomedia/neomedia.manifest.mf +++ b/src/net/java/sip/communicator/impl/neomedia/neomedia.manifest.mf @@ -25,9 +25,12 @@ Import-Package: apple.awt, javax.xml.transform.dom, javax.xml.transform.stream, net.java.sip.communicator.service.gui, + net.java.sip.communicator.service.notification, net.java.sip.communicator.service.protocol, net.java.sip.communicator.service.protocol.event, net.java.sip.communicator.service.resources, + net.java.sip.communicator.service.systray, + net.java.sip.communicator.service.systray.event, net.java.sip.communicator.util, net.java.sip.communicator.util.swing, org.bouncycastle.crypto, diff --git a/src/net/java/sip/communicator/impl/notification/PopupMessageNotificationHandlerImpl.java b/src/net/java/sip/communicator/impl/notification/PopupMessageNotificationHandlerImpl.java index c6a8ab9..3395032 100644 --- a/src/net/java/sip/communicator/impl/notification/PopupMessageNotificationHandlerImpl.java +++ b/src/net/java/sip/communicator/impl/notification/PopupMessageNotificationHandlerImpl.java @@ -8,6 +8,7 @@ package net.java.sip.communicator.impl.notification; import net.java.sip.communicator.service.notification.*; import net.java.sip.communicator.service.systray.*; +import net.java.sip.communicator.service.systray.event.*; import net.java.sip.communicator.util.Logger; import org.jitsi.util.*; @@ -62,4 +63,34 @@ public class PopupMessageNotificationHandlerImpl logger.error("Message is null or empty!", new Throwable("Null or empty message")); } + + /** + * Adds a listener for <tt>SystrayPopupMessageEvent</tt>s posted when user + * clicks on the system tray popup message. + * + * @param listener the listener to add + */ + public void addPopupMessageListener(SystrayPopupMessageListener listener) + { + SystrayService systray = NotificationActivator.getSystray(); + if(systray == null) + return; + + systray.addPopupMessageListener(listener); + } + + /** + * Removes a listener previously added with + * <tt>addPopupMessageListener</tt>. + * + * @param listener the listener to remove + */ + public void removePopupMessageListener(SystrayPopupMessageListener listener) + { + SystrayService systray = NotificationActivator.getSystray(); + if(systray == null) + return; + + systray.removePopupMessageListener(listener); + } } diff --git a/src/net/java/sip/communicator/impl/notification/notification.manifest.mf b/src/net/java/sip/communicator/impl/notification/notification.manifest.mf index 7847de3..bf3dfa4 100644 --- a/src/net/java/sip/communicator/impl/notification/notification.manifest.mf +++ b/src/net/java/sip/communicator/impl/notification/notification.manifest.mf @@ -18,4 +18,5 @@ Import-Package: org.osgi.framework, net.java.sip.communicator.service.protocol.event, org.jitsi.service.resources, net.java.sip.communicator.service.resources, - net.java.sip.communicator.service.systray + net.java.sip.communicator.service.systray, + net.java.sip.communicator.service.systray.event diff --git a/src/net/java/sip/communicator/service/notification/PopupMessageNotificationHandler.java b/src/net/java/sip/communicator/service/notification/PopupMessageNotificationHandler.java index 6327afc..e8b5716 100644 --- a/src/net/java/sip/communicator/service/notification/PopupMessageNotificationHandler.java +++ b/src/net/java/sip/communicator/service/notification/PopupMessageNotificationHandler.java @@ -6,6 +6,8 @@ */ package net.java.sip.communicator.service.notification; +import net.java.sip.communicator.service.systray.event.*; + /** * The <tt>PopupMessageNotificationHandler</tt> interface is meant to be * implemented by the notification bundle in order to provide handling of @@ -32,4 +34,21 @@ public interface PopupMessageNotificationHandler String message, byte[] icon, Object tag); + + /** + * Adds a listener for <tt>SystrayPopupMessageEvent</tt>s posted when user + * clicks on the system tray popup message. + * + * @param listener the listener to add + */ + public void addPopupMessageListener(SystrayPopupMessageListener listener); + + /** + * Removes a listener previously added with + * <tt>addPopupMessageListener</tt>. + * + * @param listener the listener to remove + */ + public void removePopupMessageListener( + SystrayPopupMessageListener listener); } diff --git a/src/net/java/sip/communicator/service/notification/notification.manifest.mf b/src/net/java/sip/communicator/service/notification/notification.manifest.mf index 6c52f21..76b85ed 100644 --- a/src/net/java/sip/communicator/service/notification/notification.manifest.mf +++ b/src/net/java/sip/communicator/service/notification/notification.manifest.mf @@ -8,4 +8,5 @@ Export-Package: net.java.sip.communicator.service.notification, net.java.sip.communicator.service.notification.event Import-Package: org.osgi.framework, net.java.sip.communicator.util, - org.jitsi.service.configuration + org.jitsi.service.configuration, + net.java.sip.communicator.service.systray.event |