|  | @@ -16,8 +16,6 @@
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  package org.ros.android;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -import com.google.common.base.Preconditions;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  |  import android.app.Activity;
 | 
	
		
			
				|  |  |  import android.content.ComponentName;
 | 
	
		
			
				|  |  |  import android.content.Intent;
 | 
	
	
		
			
				|  | @@ -25,6 +23,8 @@ import android.content.ServiceConnection;
 | 
	
		
			
				|  |  |  import android.os.AsyncTask;
 | 
	
		
			
				|  |  |  import android.os.IBinder;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +import com.google.common.base.Preconditions;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |  import org.ros.address.InetAddressFactory;
 | 
	
		
			
				|  |  |  import org.ros.exception.RosRuntimeException;
 | 
	
		
			
				|  |  |  import org.ros.node.NodeMain;
 | 
	
	
		
			
				|  | @@ -40,14 +40,64 @@ import java.net.URISyntaxException;
 | 
	
		
			
				|  |  |   */
 | 
	
		
			
				|  |  |  public abstract class RosActivity extends Activity {
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -  private static final int MASTER_CHOOSER_REQUEST_CODE = 0;
 | 
	
		
			
				|  |  | +  protected static final int MASTER_CHOOSER_REQUEST_CODE = 0;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    private final NodeMainExecutorServiceConnection nodeMainExecutorServiceConnection;
 | 
	
		
			
				|  |  |    private final String notificationTicker;
 | 
	
		
			
				|  |  |    private final String notificationTitle;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | +  private Class<?> masterChooserActivity = MasterChooser.class;
 | 
	
		
			
				|  |  | +  private int masterChooserRequestCode = MASTER_CHOOSER_REQUEST_CODE;
 | 
	
		
			
				|  |  |    protected NodeMainExecutorService nodeMainExecutorService;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +  /**
 | 
	
		
			
				|  |  | +   * Default Activity Result callback - compatible with standard {@link MasterChooser}
 | 
	
		
			
				|  |  | +   */
 | 
	
		
			
				|  |  | +  private OnActivityResultCallback onActivityResultCallback = new OnActivityResultCallback() {
 | 
	
		
			
				|  |  | +    @Override
 | 
	
		
			
				|  |  | +    public void execute(int requestCode, int resultCode, Intent data) {
 | 
	
		
			
				|  |  | +      if (resultCode == RESULT_OK) {
 | 
	
		
			
				|  |  | +        if (requestCode == MASTER_CHOOSER_REQUEST_CODE) {
 | 
	
		
			
				|  |  | +          String host;
 | 
	
		
			
				|  |  | +          String networkInterfaceName = data.getStringExtra("ROS_MASTER_NETWORK_INTERFACE");
 | 
	
		
			
				|  |  | +          // Handles the default selection and prevents possible errors
 | 
	
		
			
				|  |  | +          if (networkInterfaceName == null || networkInterfaceName.equals("")) {
 | 
	
		
			
				|  |  | +            host = getDefaultHostAddress();
 | 
	
		
			
				|  |  | +          } else {
 | 
	
		
			
				|  |  | +            try {
 | 
	
		
			
				|  |  | +              NetworkInterface networkInterface = NetworkInterface.getByName(networkInterfaceName);
 | 
	
		
			
				|  |  | +              host = InetAddressFactory.newNonLoopbackForNetworkInterface(networkInterface).getHostAddress();
 | 
	
		
			
				|  |  | +            } catch (SocketException e) {
 | 
	
		
			
				|  |  | +              throw new RosRuntimeException(e);
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +          }
 | 
	
		
			
				|  |  | +          nodeMainExecutorService.setRosHostname(host);
 | 
	
		
			
				|  |  | +          if (data.getBooleanExtra("ROS_MASTER_CREATE_NEW", false)) {
 | 
	
		
			
				|  |  | +            nodeMainExecutorService.startMaster(data.getBooleanExtra("ROS_MASTER_PRIVATE", true));
 | 
	
		
			
				|  |  | +          } else {
 | 
	
		
			
				|  |  | +            URI uri;
 | 
	
		
			
				|  |  | +            try {
 | 
	
		
			
				|  |  | +              uri = new URI(data.getStringExtra("ROS_MASTER_URI"));
 | 
	
		
			
				|  |  | +            } catch (URISyntaxException e) {
 | 
	
		
			
				|  |  | +              throw new RosRuntimeException(e);
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +            nodeMainExecutorService.setMasterUri(uri);
 | 
	
		
			
				|  |  | +          }
 | 
	
		
			
				|  |  | +          // Run init() in a new thread as a convenience since it often requires network access.
 | 
	
		
			
				|  |  | +          new AsyncTask<Void, Void, Void>() {
 | 
	
		
			
				|  |  | +            @Override
 | 
	
		
			
				|  |  | +            protected Void doInBackground(Void... params) {
 | 
	
		
			
				|  |  | +              RosActivity.this.init(nodeMainExecutorService);
 | 
	
		
			
				|  |  | +              return null;
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +          }.execute();
 | 
	
		
			
				|  |  | +        } else {
 | 
	
		
			
				|  |  | +          // Without a master URI configured, we are in an unusable state.
 | 
	
		
			
				|  |  | +          nodeMainExecutorService.forceShutdown();
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +      }
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +  };
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |    private final class NodeMainExecutorServiceConnection implements ServiceConnection {
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      private NodeMainExecutorServiceListener serviceListener;
 | 
	
	
		
			
				|  | @@ -98,10 +148,23 @@ public abstract class RosActivity extends Activity {
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    };
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +  /**
 | 
	
		
			
				|  |  | +   * Standard constructor.
 | 
	
		
			
				|  |  | +   * Use this constructor to proceed using the standard {@link MasterChooser}.
 | 
	
		
			
				|  |  | +   * @param notificationTicker Title to use in Ticker notifications.
 | 
	
		
			
				|  |  | +   * @param notificationTitle Title to use in notifications.
 | 
	
		
			
				|  |  | +     */
 | 
	
		
			
				|  |  |    protected RosActivity(String notificationTicker, String notificationTitle) {
 | 
	
		
			
				|  |  |      this(notificationTicker, notificationTitle, null);
 | 
	
		
			
				|  |  |    }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +  /**
 | 
	
		
			
				|  |  | +   * Custom Master URI constructor.
 | 
	
		
			
				|  |  | +   * Use this constructor to skip launching {@link MasterChooser}.
 | 
	
		
			
				|  |  | +   * @param notificationTicker Title to use in Ticker notifications.
 | 
	
		
			
				|  |  | +   * @param notificationTitle Title to use in notifications.
 | 
	
		
			
				|  |  | +   * @param customMasterUri URI of the ROS master to connect to.
 | 
	
		
			
				|  |  | +     */
 | 
	
		
			
				|  |  |    protected RosActivity(String notificationTicker, String notificationTitle, URI customMasterUri) {
 | 
	
		
			
				|  |  |      super();
 | 
	
		
			
				|  |  |      this.notificationTicker = notificationTicker;
 | 
	
	
		
			
				|  | @@ -109,6 +172,23 @@ public abstract class RosActivity extends Activity {
 | 
	
		
			
				|  |  |      nodeMainExecutorServiceConnection = new NodeMainExecutorServiceConnection(customMasterUri);
 | 
	
		
			
				|  |  |    }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +  /**
 | 
	
		
			
				|  |  | +   * Custom MasterChooser constructor.
 | 
	
		
			
				|  |  | +   * Use this constructor to specify which {@link Activity} should be started in place of {@link MasterChooser}.
 | 
	
		
			
				|  |  | +   * The specified activity shall return a result that can be handled by a custom callback.
 | 
	
		
			
				|  |  | +   * See {@link #setOnActivityResultCallback(OnActivityResultCallback)} for more information about
 | 
	
		
			
				|  |  | +   * how to handle custom request codes and results.
 | 
	
		
			
				|  |  | +   * @param notificationTicker Title to use in Ticker notifications.
 | 
	
		
			
				|  |  | +   * @param notificationTitle Title to use in notifications.
 | 
	
		
			
				|  |  | +   * @param activity {@link Activity} to launch instead of {@link MasterChooser}.
 | 
	
		
			
				|  |  | +   * @param requestCode Request identifier to start the given {@link Activity} for a result.
 | 
	
		
			
				|  |  | +     */
 | 
	
		
			
				|  |  | +  protected RosActivity(String notificationTicker, String notificationTitle, Class<?> activity, int requestCode) {
 | 
	
		
			
				|  |  | +    this(notificationTicker, notificationTitle);
 | 
	
		
			
				|  |  | +    masterChooserActivity = activity;
 | 
	
		
			
				|  |  | +    masterChooserRequestCode = requestCode;
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |    @Override
 | 
	
		
			
				|  |  |    protected void onStart() {
 | 
	
		
			
				|  |  |      super.onStart();
 | 
	
	
		
			
				|  | @@ -161,7 +241,7 @@ public abstract class RosActivity extends Activity {
 | 
	
		
			
				|  |  |      Preconditions.checkState(getMasterUri() == null);
 | 
	
		
			
				|  |  |      // Call this method on super to avoid triggering our precondition in the
 | 
	
		
			
				|  |  |      // overridden startActivityForResult().
 | 
	
		
			
				|  |  | -    super.startActivityForResult(new Intent(this, MasterChooser.class), 0);
 | 
	
		
			
				|  |  | +    super.startActivityForResult(new Intent(this, masterChooserActivity), masterChooserRequestCode);
 | 
	
		
			
				|  |  |    }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    public URI getMasterUri() {
 | 
	
	
		
			
				|  | @@ -176,57 +256,33 @@ public abstract class RosActivity extends Activity {
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    @Override
 | 
	
		
			
				|  |  |    public void startActivityForResult(Intent intent, int requestCode) {
 | 
	
		
			
				|  |  | -    Preconditions.checkArgument(requestCode != MASTER_CHOOSER_REQUEST_CODE);
 | 
	
		
			
				|  |  | +    Preconditions.checkArgument(requestCode != masterChooserRequestCode);
 | 
	
		
			
				|  |  |      super.startActivityForResult(intent, requestCode);
 | 
	
		
			
				|  |  |    }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    @Override
 | 
	
		
			
				|  |  |    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
 | 
	
		
			
				|  |  |      super.onActivityResult(requestCode, resultCode, data);
 | 
	
		
			
				|  |  | -    if (resultCode == RESULT_OK) {
 | 
	
		
			
				|  |  | -      if (requestCode == MASTER_CHOOSER_REQUEST_CODE) {
 | 
	
		
			
				|  |  | -        String host;
 | 
	
		
			
				|  |  | -        String networkInterfaceName = data.getStringExtra("ROS_MASTER_NETWORK_INTERFACE");
 | 
	
		
			
				|  |  | -        // Handles the default selection and prevents possible errors
 | 
	
		
			
				|  |  | -        if (networkInterfaceName == null || networkInterfaceName.equals("")) {
 | 
	
		
			
				|  |  | -          host = getDefaultHostAddress();
 | 
	
		
			
				|  |  | -        } else {
 | 
	
		
			
				|  |  | -          try {
 | 
	
		
			
				|  |  | -            NetworkInterface networkInterface = NetworkInterface.getByName(networkInterfaceName);
 | 
	
		
			
				|  |  | -            host = InetAddressFactory.newNonLoopbackForNetworkInterface(networkInterface).getHostAddress();
 | 
	
		
			
				|  |  | -          } catch (SocketException e) {
 | 
	
		
			
				|  |  | -            throw new RosRuntimeException(e);
 | 
	
		
			
				|  |  | -          }
 | 
	
		
			
				|  |  | -        }
 | 
	
		
			
				|  |  | -        nodeMainExecutorService.setRosHostname(host);
 | 
	
		
			
				|  |  | -        if (data.getBooleanExtra("ROS_MASTER_CREATE_NEW", false)) {
 | 
	
		
			
				|  |  | -          nodeMainExecutorService.startMaster(data.getBooleanExtra("ROS_MASTER_PRIVATE", true));
 | 
	
		
			
				|  |  | -        } else {
 | 
	
		
			
				|  |  | -          URI uri;
 | 
	
		
			
				|  |  | -          try {
 | 
	
		
			
				|  |  | -            uri = new URI(data.getStringExtra("ROS_MASTER_URI"));
 | 
	
		
			
				|  |  | -          } catch (URISyntaxException e) {
 | 
	
		
			
				|  |  | -            throw new RosRuntimeException(e);
 | 
	
		
			
				|  |  | -          }
 | 
	
		
			
				|  |  | -          nodeMainExecutorService.setMasterUri(uri);
 | 
	
		
			
				|  |  | -        }
 | 
	
		
			
				|  |  | -        // Run init() in a new thread as a convenience since it often requires network access.
 | 
	
		
			
				|  |  | -        new AsyncTask<Void, Void, Void>() {
 | 
	
		
			
				|  |  | -          @Override
 | 
	
		
			
				|  |  | -          protected Void doInBackground(Void... params) {
 | 
	
		
			
				|  |  | -            RosActivity.this.init(nodeMainExecutorService);
 | 
	
		
			
				|  |  | -            return null;
 | 
	
		
			
				|  |  | -          }
 | 
	
		
			
				|  |  | -        }.execute();
 | 
	
		
			
				|  |  | -      } else {
 | 
	
		
			
				|  |  | -        // Without a master URI configured, we are in an unusable state.
 | 
	
		
			
				|  |  | -        nodeMainExecutorService.forceShutdown();
 | 
	
		
			
				|  |  | -      }
 | 
	
		
			
				|  |  | +    if (onActivityResultCallback != null) {
 | 
	
		
			
				|  |  | +      onActivityResultCallback.execute(requestCode, resultCode, data);
 | 
	
		
			
				|  |  |      }
 | 
	
		
			
				|  |  | -    super.onActivityResult(requestCode, resultCode, data);
 | 
	
		
			
				|  |  |    }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    private String getDefaultHostAddress() {
 | 
	
		
			
				|  |  |      return InetAddressFactory.newNonLoopback().getHostAddress();
 | 
	
		
			
				|  |  |    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  public interface OnActivityResultCallback {
 | 
	
		
			
				|  |  | +    void execute(int requestCode, int resultCode, Intent data);
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  /**
 | 
	
		
			
				|  |  | +   * Set a callback that will be called onActivityResult.
 | 
	
		
			
				|  |  | +   * Custom callbacks should be able to handle custom request codes configured
 | 
	
		
			
				|  |  | +   * in custom Activity constructor {@link #RosActivity(String, String, Class, int)}.
 | 
	
		
			
				|  |  | +   * @param callback Action that will be performed when this Activity gets a result.
 | 
	
		
			
				|  |  | +     */
 | 
	
		
			
				|  |  | +  public void setOnActivityResultCallback(OnActivityResultCallback callback) {
 | 
	
		
			
				|  |  | +    onActivityResultCallback = callback;
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  |  }
 |