package org.multibit.hd.ui.events.view; import com.google.common.base.Optional; import com.google.common.base.Preconditions; import com.google.common.collect.Sets; import com.google.common.eventbus.EventBus; import org.bitcoinj.core.Coin; import org.multibit.hd.core.dto.RAGStatus; import org.multibit.hd.core.error_reporting.ExceptionHandler; import org.multibit.hd.ui.events.controller.ShowScreenEvent; import org.multibit.hd.ui.models.AlertModel; import org.multibit.hd.ui.views.ViewKey; import org.multibit.hd.ui.views.components.Panels; import org.multibit.hd.ui.views.components.wallet_detail.WalletDetail; import org.multibit.hd.ui.views.screens.Screen; import org.multibit.hd.ui.views.wizards.AbstractWizardModel; import org.multibit.hd.ui.views.wizards.WizardButton; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import javax.swing.*; import java.math.BigDecimal; import java.util.Set; /** * <p>Factory to provide the following to application API:</p> * <ul> * <li>Entry point to broadcast application events associated with the UI</li> * </ul> * <p>An application event is a high level event with specific semantics. Normally a * low level event (such as a mouse click) will initiate it.</p> * <p/> * <p>It is expected that ViewEvents will interact with Swing components and as such is * expected to execute on the EDT. This cannot be provided directly within the method * by wrapping since the semantics of the calling code may require synchronous execution * across many subscribers. One example is if the UI is required to "freeze" in order to * prevent the user from interacting with it during an atomic operation.</p> * * @since 0.0.1 */ public class ViewEvents { private static final Logger log = LoggerFactory.getLogger(ViewEvents.class); /** * Use Guava to handle subscribers to events * Do not use this method directly, instead */ private static final EventBus viewEventBus = new EventBus(ExceptionHandler.newSubscriberExceptionHandler()); /** * Keep track of the Guava event bus subscribers for a clean shutdown */ private static final Set<Object> viewEventBusSubscribers = Sets.newHashSet(); /** * Utilities have a private constructor */ private ViewEvents() { } /** * <p>Subscribe to events. Repeating a subscribe will not affect the event bus.</p> * <p>This approach ensures all subscribers will be correctly removed during a shutdown or wizard hide event</p> * * @param subscriber The subscriber (use the Guava <code>@Subscribe</code> annotation to subscribe a method) */ public static void subscribe(Object subscriber) { Preconditions.checkNotNull(subscriber, "'subscriber' must be present"); if (viewEventBusSubscribers.add(subscriber)) { log.trace("Register: " + subscriber.getClass().getSimpleName()); try { viewEventBus.register(subscriber); } catch (IllegalArgumentException e) { log.warn("Unexpected failure to register"); } } else { log.warn("Subscriber already registered: " + subscriber.getClass().getSimpleName()); } } /** * <p>Unsubscribe a known subscriber from events. Providing an unknown object will not affect the event bus.</p> * <p>This approach ensures all subscribers will be correctly removed during a shutdown or wizard hide event</p> * * @param subscriber The subscriber (use the Guava <code>@Subscribe</code> annotation to subscribe a method) */ public static void unsubscribe(Object subscriber) { Preconditions.checkNotNull(subscriber, "'subscriber' must be present"); if (viewEventBusSubscribers.contains(subscriber)) { log.trace("Unregister: " + subscriber.getClass().getSimpleName()); try { viewEventBus.unregister(subscriber); } catch (IllegalArgumentException e) { log.warn("Unexpected failure to unregister"); } viewEventBusSubscribers.remove(subscriber); } } /** * <p>Unsubscribe all subscribers from events</p> * <p>This approach ensures all subscribers will be correctly removed during a shutdown or wizard hide event</p> */ @SuppressWarnings("unchecked") public static void unsubscribeAll() { Set allSubscribers = Sets.newHashSet(); allSubscribers.addAll(viewEventBusSubscribers); for (Object subscriber : allSubscribers) { unsubscribe(subscriber); } allSubscribers.clear(); log.info("All subscribers removed"); } /** * <p>Broadcast a new "balance changed" event</p> * * @param coinBalance The current balance in coins * @param localBalance The current balance in local currency * @param rateProvider The exchange rate provider (e.g. "Bitstamp") */ public static void fireBalanceChangedEvent( final Coin coinBalance, final Coin coinWithUnconfirmedBalance, final BigDecimal localBalance, final Optional<String> rateProvider ) { log.trace("Firing 'balance changed' event"); if (SwingUtilities.isEventDispatchThread()) { viewEventBus.post( new BalanceChangedEvent( coinBalance, coinWithUnconfirmedBalance, localBalance, rateProvider )); } else { SwingUtilities.invokeLater(new Runnable() { @Override public void run() { viewEventBus.post( new BalanceChangedEvent( coinBalance, coinWithUnconfirmedBalance, localBalance, rateProvider )); } }); } } /** * <p>Broadcast a new "system status changed" event</p> * * @param localisedMessage The localised message to display alongside the severity * @param severity The system status severity (normally in line with an alert) */ public static void fireSystemStatusChangedEvent(final String localisedMessage, final RAGStatus severity) { log.trace("Firing 'system status changed' event"); if (SwingUtilities.isEventDispatchThread()) { viewEventBus.post(new SystemStatusChangedEvent(localisedMessage, severity)); } else { SwingUtilities.invokeLater(new Runnable() { @Override public void run() { viewEventBus.post(new SystemStatusChangedEvent(localisedMessage, severity)); } }); } } /** * <p>Broadcast a new "progress changed" event </p> * * @param localisedMessage The localised message to display alongside the progress bar * @param percent The amount to display in percent */ public static void fireProgressChangedEvent(final String localisedMessage, final int percent) { log.trace("Firing 'progress changed' event: '{}'", percent); if (SwingUtilities.isEventDispatchThread()) { viewEventBus.post(new ProgressChangedEvent(localisedMessage, percent)); } else { SwingUtilities.invokeLater(new Runnable() { @Override public void run() { viewEventBus.post(new ProgressChangedEvent(localisedMessage, percent)); } }); } } /** * <p>Broadcast a new "alert added" event</p> * * @param alertModel The alert model for the new display */ public static void fireAlertAddedEvent(final AlertModel alertModel) { log.trace("Firing 'alert added' event"); if (SwingUtilities.isEventDispatchThread()) { viewEventBus.post(new AlertAddedEvent(alertModel)); } else { SwingUtilities.invokeLater(new Runnable() { @Override public void run() { viewEventBus.post(new AlertAddedEvent(alertModel)); } }); } } /** * <p>Broadcast a new "switch wallet" event</p> */ public static void fireSwitchWalletEvent() { log.debug("Firing 'switch wallet' event"); if (SwingUtilities.isEventDispatchThread()) { viewEventBus.post(new SwitchWalletEvent()); } else { SwingUtilities.invokeLater(new Runnable() { @Override public void run() { viewEventBus.post(new SwitchWalletEvent()); } }); } } /** * <p>Broadcast a new "alert removed" event</p> */ public static void fireAlertRemovedEvent() { log.trace("Firing 'alert removed' event"); if (SwingUtilities.isEventDispatchThread()) { viewEventBus.post(new AlertRemovedEvent()); } else { SwingUtilities.invokeLater(new Runnable() { @Override public void run() { viewEventBus.post(new AlertRemovedEvent()); } }); } } /** * <p>Broadcast a new "wallet detail changed" event</p> */ public static void fireWalletDetailChangedEvent(final WalletDetail walletDetail) { log.trace("Firing 'walletDetailChanged' event"); if (SwingUtilities.isEventDispatchThread()) { viewEventBus.post(new WalletDetailChangedEvent(walletDetail)); } else { SwingUtilities.invokeLater(new Runnable() { @Override public void run() { viewEventBus.post(new WalletDetailChangedEvent(walletDetail)); } }); } } /** * <p>Broadcast a new "wizard button enabled" event</p> * * @param panelName The panel name to which this applies * @param wizardButton The wizard button to which this applies * @param enabled True if the button should be enabled */ public static void fireWizardButtonEnabledEvent( final String panelName, final WizardButton wizardButton, final boolean enabled ) { log.trace("Firing 'wizard button enabled {}' event: {}", panelName, enabled); if (SwingUtilities.isEventDispatchThread()) { viewEventBus.post(new WizardButtonEnabledEvent(panelName, wizardButton, enabled)); } else { SwingUtilities.invokeLater(new Runnable() { @Override public void run() { viewEventBus.post(new WizardButtonEnabledEvent(panelName, wizardButton, enabled)); } }); } } /** * <p>Broadcast a new "wizard hide" event</p> * * @param panelName The unique panel name to which this applies (use screen name for detail screens) * @param wizardModel The wizard model containing all the user data * @param isExitCancel True if this hide event comes as a result of an exit or cancel */ public static void fireWizardHideEvent( final String panelName, final AbstractWizardModel wizardModel, final boolean isExitCancel ) { log.trace("Firing 'wizard hide' event"); if (SwingUtilities.isEventDispatchThread()) { viewEventBus.post(new WizardHideEvent(panelName, wizardModel, isExitCancel)); } else { SwingUtilities.invokeLater(new Runnable() { @Override public void run() { viewEventBus.post(new WizardHideEvent(panelName, wizardModel, isExitCancel)); } }); } } /** * <p>Broadcast a new "wizard popover hide" event</p> * * @param panelName The unique panel name to which this applies (use screen name for detail screens) * @param isExitCancel True if this hide event comes as a result of an exit or cancel */ public static void fireWizardPopoverHideEvent(final String panelName, final boolean isExitCancel) { log.trace("Firing 'wizard popover hide' event"); if (SwingUtilities.isEventDispatchThread()) { viewEventBus.post(new WizardPopoverHideEvent(panelName, isExitCancel)); } else { SwingUtilities.invokeLater(new Runnable() { @Override public void run() { viewEventBus.post(new WizardPopoverHideEvent(panelName, isExitCancel)); } }); } } /** * <p>Broadcast a new "wizard deferred hide" event</p> * * @param panelName The unique panel name to which this applies (use screen name for detail screens) * @param isExitCancel True if this deferred hide event comes as a result of an exit or cancel */ public static void fireWizardDeferredHideEvent(final String panelName, final boolean isExitCancel) { log.trace("Firing 'wizard deferred hide' event"); // Prevent any light box creation activity ahead of this process Panels.setDeferredHideEventInProgress(true); if (SwingUtilities.isEventDispatchThread()) { viewEventBus.post(new WizardDeferredHideEvent(panelName, isExitCancel)); } else { SwingUtilities.invokeLater(new Runnable() { @Override public void run() { viewEventBus.post(new WizardDeferredHideEvent(panelName, isExitCancel)); } }); } } /** * <p>Broadcast a new "component changed" event</p> * * @param panelName The unique panel name to which this applies (use screen name for detail screens) * @param componentModel The component model containing the change (absent if the component has no model) */ public static void fireComponentChangedEvent(final String panelName, final Optional componentModel) { log.trace("Firing 'component changed' event"); if (SwingUtilities.isEventDispatchThread()) { viewEventBus.post(new ComponentChangedEvent(panelName, componentModel)); } else { SwingUtilities.invokeLater(new Runnable() { @Override public void run() { viewEventBus.post(new ComponentChangedEvent(panelName, componentModel)); } }); } } /** * <p>Broadcast a new "verification status changed" event</p> * * @param panelName The panel name to which this applies * @param status True if the verification is OK */ public static void fireVerificationStatusChangedEvent(final String panelName, final boolean status) { log.trace("Firing 'verification status changed' event: {}", status); if (SwingUtilities.isEventDispatchThread()) { viewEventBus.post(new VerificationStatusChangedEvent(panelName, status)); } else { SwingUtilities.invokeLater(new Runnable() { @Override public void run() { viewEventBus.post(new VerificationStatusChangedEvent(panelName, status)); } }); } } /** * <p>Broadcast a new "view changed" event</p> * * @param viewKey The view to which this applies * @param visible True if the view is "visible" (could be reduced height etc) */ public static void fireViewChangedEvent(final ViewKey viewKey, final boolean visible) { log.trace("Firing 'view changed' event: {}", visible); if (SwingUtilities.isEventDispatchThread()) { viewEventBus.post(new ViewChangedEvent(viewKey, visible)); } else { SwingUtilities.invokeLater(new Runnable() { @Override public void run() { viewEventBus.post(new ViewChangedEvent(viewKey, visible)); } }); } } /** * <p>Broadcast a new "show detail screen" event</p> * * @param detailScreen The screen to show */ public static void fireShowDetailScreenEvent(final Screen detailScreen) { log.trace("Firing 'show detail screen' event"); if (SwingUtilities.isEventDispatchThread()) { viewEventBus.post(new ShowScreenEvent(detailScreen)); } else { SwingUtilities.invokeLater(new Runnable() { @Override public void run() { viewEventBus.post(new ShowScreenEvent(detailScreen)); } }); } } }