Mobile Programming Lecture 9 Bound Service, Location, Sensors, IntentFilter.

Slides:



Advertisements
Similar presentations
Cosc 5/4730 Android Services. What is a service? From android developer web pages: Most confusion about the Service class actually revolves around what.
Advertisements

Android 02: Activities David Meredith
All About Android Introduction to Android 1. Creating a New App “These aren’t the droids we’re looking for.” Obi-wan Kenobi 1. Bring up Eclipse. 2. Click.
The Android Activity Lifecycle. Slide 2 Introduction Working with the Android logging system Rotation and multiple layouts Understanding the Android activity.
Manifest File, Intents, and Multiple Activities. Manifest File.
The Activity Class 1.  One application component type  Provides a visual interface for a single screen  Typically supports one thing a user can do,
Application Fundamentals. See: developer.android.com/guide/developing/building/index.html.
Location & Maps.  Mobile applications can benefit from being location-aware, e.g.,  Routing from a current to a desired location  Searching for stores.
Cosc 4730 Android TabActivity and ListView. TabActivity A TabActivity allows for multiple “tabs”. – Each Tab is it’s own activity and the “root” activity.
Android Sensors & Async Callbacks Jules White Bradley Dept. of Electrical and Computer Engineering Virginia Tech
Programming Mobile Applications with Android September, Albacete, Spain Jesus Martínez-Gómez.
Getting Started with Android APIs Ivan Wong. Motivation - “Datasheet” - Recently exposed to what’s available in Android - So let’s see what API’s are.
 Understanding an activity  Starting an activity  Passing information between activities  Understanding intents  Understanding the activity lifecycle.
ANDROID UI – FRAGMENTS. Fragment  An activity is a container for views  When you have a larger screen device than a phone –like a tablet it can look.
Software Architecture of Android Yaodong Bi, Ph.D. Department of Computing Sciences University of Scranton.
1 Localization and Sensing Nilanjan Banerjee Mobile Systems Programming (Acknowledgement: Jules White) University of Arkansas Fayetteville, AR
Mobile Programming Lecture 16 The Facebook API. Agenda The Setup Hello, Facebook User Facebook Permissions Access Token Logging Out Graph API.
Cosc 5/4730 Introduction: Threads, Android Activities, and MVC.
Rajab Davudov. Agenda Eclipse, ADT and Android SDK APK file Fundamentals – Activity – Service – Content Provider – Broadcast Receiver – Intent Hello World.
Cosc 5/4730 Broadcast Receiver. Broadcast receiver A broadcast receiver (short receiver) – is an Android component which allows you to register for system.
1 Announcements Homework #2 due Feb 7 at 1:30pm Submit the entire Eclipse project in Blackboard Please fill out the when2meets when your Project Manager.
Android Accessing GPS Ken Nguyen Clayton State University 2012.
1 CMSC 628: Introduction to Mobile Computing Nilanjan Banerjee Introduction to Mobile Computing University of Maryland Baltimore County
Copyright© Jeffrey Jongko, Ateneo de Manila University Of Activities, Intents and Applications.
Chapter 2 The Android User Interface. Objectives  In this chapter, you learn to:  Develop a user interface using the TextView, ImageView, and Button.
CS378 - Mobile Computing Intents. Allow us to use applications and components that are part of Android System – start activities – start services – deliver.
Cosc 5/4730 Android Communications Intents, callbacks, and setters.
Services A Service is an application component that can perform long-running operations in the background and does not provide a user interface. An application.
Location. GPS Global Positioning System – At least 4 satellites typically used 3 required extra for error detection and altitude typically accurate within.
Maps Dr. David Janzen Except as otherwise noted, the content of this presentation is licensed under the Creative Commons Attribution 2.5 License.
Mobile Programming Midterm Review
Mobile Programming Lecture 3 Debugging. Lecture 2 Review What widget would you use to allow the user to enter o a yes/no value o a range of values from.
Applications with Multiple Activities. Most applications will have more than one activity. The main activity is started when the application is started.
Android - Location Based Services. Google Play services facilitates adding location awareness to your app with automated location tracking Geo fencing.
Working with Multiple Activities. Slide 2 Introduction Working with multiple activities Creating multiple views Introduction to intents Passing data to.
Introducing Intents Intents Bind application components and navigate between them Transform device into collection of interconnected systems Creating a.
Sensors – Part 2 SE 395/595. Location in Android LocationManager class – Configure providers and their listeners LocationListener class – Handles update.
Services Background operating component without a visual interface Running in the background indefinitely Differently from Activity, Service in Android.
Lecture 2: Android Concepts
Lecture 4: Sensors Topics: Motion, Position, and Environmental Sensors Date: Feb 11, 2016.
Activities and Intents Chapter 3 1. Objectives Explore an activity’s lifecycle Learn about saving and restoring an activity Understand intents and how.
Mobile Programming Lecture 4 Resources, Selection, Activities, Intents.
Intents and Broadcast Receivers Dr. David Janzen Except as otherwise noted, the content of this presentation is licensed under the Creative Commons Attribution.
Lecture 5: Location Topics: Google Play Services, Location API Date: Feb 16, 2016.
Working with Multiple Activities. Slide 2 Introduction Working with multiple activities Putting together the AndroidManifest.xml file Creating multiple.
1. 2 Android location services Determining a device’s current location Tracking device movements Proximity alerts.
CS499 – Mobile Application Development Fall 2013 Location & Maps.
Android Android Sensors Android Sensors: – Accelerometer – Gravity sensor – Linear Acceleration sensor – Magnetic Field sensor – Orientation.
Sensors in Android.
Lecture 4: Sensors Topics: Motion, Position, and Environmental Sensors
Intents and Broadcast Receivers
Android – Event Handling
Lecture 4: Sensors Topics: Motion, Position, and Environmental Sensors.
Activities and Intents
Sensors, maps and fragments:
CS499 – Mobile Application Development
CIS 470 Mobile App Development
Activities and Intents
Activities and Intents
CIS 470 Mobile App Development
Lecture 5: Location Topics: Google Play Services, Location API.
Android Developer Fundamentals V2 Lesson 5
Mobile Programming Sensors in Android.
SE4S701 Mobile Application Development
Activities and Fragments
Activities, Fragments, and Intents
CIS 470 Mobile App Development
Mobile Programming Broadcast Receivers.
CIS 694/EEC 693 Android Sensor Programming
CIS 694/EEC 693 Android Sensor Programming
Presentation transcript:

Mobile Programming Lecture 9 Bound Service, Location, Sensors, IntentFilter

Agenda Bound Services Location Sensor Starting an Activity for a Result Understanding Implicit Intents

Bound Service When you create a Bound Service, you must provide an IBinder that provides the programming interface that clients can use to interact with the Service There are 3 ways that you can provide this interface: 1. Extending the Binder class 2. Using a Messenger 3. Using AIDL

Bound Service

Bound Service - 1 extending Binder this is the preferred technique when your service is merely a background worker for your own application the only reason you would not create your interface this way is because your service is used by other applications or across separate processes

Bound Service - 1 extending Binder Here's how to set it up 1.In your service, create an instance of Binder that returns the current Service instance, which has public methods the client can callBinderService 2.Return this instance of Binder from the onBind() callback methodBinderonBind() 3.In the client, receive the Binder from the onServiceConnected() callback method and make calls to the bound service using the methods providedBinder onServiceConnected()

Bound Service - 1 extending Binder see BoundServiceBinderExample.tar ServiceConnection class monitors the connection between the client and the Service When a Service is bound, the binder returned is returned to the onServiceConnected() method of ServiceConnection

Bound Service - 2 using Messenger if you need your interface to work across different processes, you can create an interface for the service with a Messenger the service defines a Handler that responds to different types of Message objects Additionally, the client can define a Messenger of its own so the service can send messages back. this is the simplest way to perform interprocess communication (IPC).

Bound Service - 2 using Messenger Notice that the handleMessage() method in the Handler is where the service receives the incoming Message and decides what to do, based on the what member.handleMessage()HandlerMessagewhat

Bound Service - 2 using Messenger see BoundServiceMessengerExample.tar

Bound Service - 3 using the AIDL Using AIDL is confusing AIDL isn't used that often We won't cover this

Location Providers To obtain user location, you can use GPS o most accurate o but  consumes battery power fast  takes a while to determine location Network Location Provider

Location - LocationManager Requesting Location Updates To get the user location, you need to use the LocationManager, which is a system service This returns a Location object, which can tell you o latitude o longitude o distance between two locations (comparing to another Location object) o accuracy of the Location in meters o direction of travel in degrees o speed of travel in meters per second The Location, however, does not give you any human readable address such as street name, state, or country

Requesting Location Updates You can use the Location object to obtain a human-readable address by using a Geocoder Geocoder can give you a list of addresses (since it may not always be sure because of accuracy issues) o returns a List of Address objects, i.e. List  street name  city  state  country  zip code Location - Geocoder and Address

Location - Last Known Location Requesting Location Updates Since GPS and Wifi location are not quite instantaeous, you can get the last known location until one of them becomes available

Location - Last Known Location final LocationManager lm = (LocationManager) getSystemService(Context.LOCATION_SERVICE); getButton = (Button) findViewById(R.id.button1); tv = (TextView) findViewById(R.id.textView1); getButton.setOnClickListener(new View.OnClickListener() public void onClick(View v) { Location lastLoc = lm.getLastKnownLocation( LocationManager.NETWORK_PROVIDER); Geocoder geo = new Geocoder(getApplicationContext()); List addresses; addresses = geo.getFromLocation(lastLoc.getLatitude(), lastLoc.getLongitude(), 1); if(addresses != null) tv.setText(addresses.get(0).getAddressLine(0)); } });

Location - Last Known Location final LocationManager lm = (LocationManager) getSystemService(Context.LOCATION_SERVICE); getButton = (Button) findViewById(R.id.button1); tv = (TextView) findViewById(R.id.textView1); getButton.setOnClickListener(new View.OnClickListener() public void onClick(View v) { Location lastLoc = lm.getLastKnownLocation( LocationManager.NETWORK_PROVIDER); Geocoder geo = new Geocoder(getApplicationContext()); List addresses; addresses = geo.getFromLocation(lastLoc.getLatitude(), lastLoc.getLongitude(), 1); if(addresses != null) tv.setText(addresses.get(0).getAddressLine(0)); } }); Similar to getting the system service for the DownloadManager

Location - Last Known Location final LocationManager lm = (LocationManager) getSystemService(Context.LOCATION_SERVICE); getButton = (Button) findViewById(R.id.button1); tv = (TextView) findViewById(R.id.textView1); getButton.setOnClickListener(new View.OnClickListener() public void onClick(View v) { Location lastLoc = lm.getLastKnownLocation( LocationManager.NETWORK_PROVIDER); Geocoder geo = new Geocoder(getApplicationContext()); List addresses; addresses = geo.getFromLocation(lastLoc.getLatitude(), lastLoc.getLongitude(), 1); if(addresses != null) tv.setText(addresses.get(0).getAddressLine(0)); } }); What was the last known location?

Location - Last Known Location final LocationManager lm = (LocationManager) getSystemService(Context.LOCATION_SERVICE); getButton = (Button) findViewById(R.id.button1); tv = (TextView) findViewById(R.id.textView1); getButton.setOnClickListener(new View.OnClickListener() public void onClick(View v) { Location lastLoc = lm.getLastKnownLocation( LocationManager.NETWORK_PROVIDER); Geocoder geo = new Geocoder(getApplicationContext()); List addresses; addresses = geo.getFromLocation(lastLoc.getLatitude(), lastLoc.getLongitude(), 1); if(addresses != null) tv.setText(addresses.get(0).getAddressLine(0)); } }); We need to use Geocoder to transform longitude and latitude to an address

Location - Last Known Location final LocationManager lm = (LocationManager) getSystemService(Context.LOCATION_SERVICE); getButton = (Button) findViewById(R.id.button1); tv = (TextView) findViewById(R.id.textView1); getButton.setOnClickListener(new View.OnClickListener() public void onClick(View v) { Location lastLoc = lm.getLastKnownLocation( LocationManager.NETWORK_PROVIDER); Geocoder geo = new Geocoder(getApplicationContext()); List addresses; addresses = geo.getFromLocation(lastLoc.getLatitude(), lastLoc.getLongitude(), 1); if(addresses != null) tv.setText(addresses.get(0).getAddressLine(0)); } }); We will store address here

Location - Last Known Location final LocationManager lm = (LocationManager) getSystemService(Context.LOCATION_SERVICE); getButton = (Button) findViewById(R.id.button1); tv = (TextView) findViewById(R.id.textView1); getButton.setOnClickListener(new View.OnClickListener() public void onClick(View v) { Location lastLoc = lm.getLastKnownLocation( LocationManager.NETWORK_PROVIDER); Geocoder geo = new Geocoder(getApplicationContext()); List addresses; addresses = geo.getFromLocation(lastLoc.getLatitude(), lastLoc.getLongitude(), 1); if(addresses != null) tv.setText(addresses.get(0).getAddressLine(0)); } }); Let's get a List of Addresses (although there may be only 1 sometimes)

Location - Last Known Location final LocationManager lm = (LocationManager) getSystemService(Context.LOCATION_SERVICE); getButton = (Button) findViewById(R.id.button1); tv = (TextView) findViewById(R.id.textView1); getButton.setOnClickListener(new View.OnClickListener() public void onClick(View v) { Location lastLoc = lm.getLastKnownLocation( LocationManager.NETWORK_PROVIDER); Geocoder geo = new Geocoder(getApplicationContext()); List addresses; addresses = geo.getFromLocation(lastLoc.getLatitude(), lastLoc.getLongitude(), 1); if(addresses != null) tv.setText(addresses.get(0).getAddressLine(0)); } }); Pass the latitude...

Location - Last Known Location final LocationManager lm = (LocationManager) getSystemService(Context.LOCATION_SERVICE); getButton = (Button) findViewById(R.id.button1); tv = (TextView) findViewById(R.id.textView1); getButton.setOnClickListener(new View.OnClickListener() public void onClick(View v) { Location lastLoc = lm.getLastKnownLocation( LocationManager.NETWORK_PROVIDER); Geocoder geo = new Geocoder(getApplicationContext()); List addresses; addresses = geo.getFromLocation(lastLoc.getLatitude(), lastLoc.getLongitude(), 1); if(addresses != null) tv.setText(addresses.get(0).getAddressLine(0)); } }); the longitude...

Location - Last Known Location final LocationManager lm = (LocationManager) getSystemService(Context.LOCATION_SERVICE); getButton = (Button) findViewById(R.id.button1); tv = (TextView) findViewById(R.id.textView1); getButton.setOnClickListener(new View.OnClickListener() public void onClick(View v) { Location lastLoc = lm.getLastKnownLocation( LocationManager.NETWORK_PROVIDER); Geocoder geo = new Geocoder(getApplicationContext()); List addresses; addresses = geo.getFromLocation(lastLoc.getLatitude(), lastLoc.getLongitude(), 1); if(addresses != null) tv.setText(addresses.get(0).getAddressLine(0)); } }); and the max number of addresses that you want to be returned

Location - Last Known Location final LocationManager lm = (LocationManager) getSystemService(Context.LOCATION_SERVICE); getButton = (Button) findViewById(R.id.button1); tv = (TextView) findViewById(R.id.textView1); getButton.setOnClickListener(new View.OnClickListener() public void onClick(View v) { Location lastLoc = lm.getLastKnownLocation( LocationManager.NETWORK_PROVIDER); Geocoder geo = new Geocoder(getApplicationContext()); List addresses; addresses = geo.getFromLocation(lastLoc.getLatitude(), lastLoc.getLongitude(), 1); if(addresses != null) tv.setText(addresses.get(0).getAddressLine(0)); } }); Get the first line of the first address in the list

Location - Last Known Location See LastKnownLocationExample.tar

Location - Location Updates Requesting Location Updates To get actual location updates periodically, you should use a LocationListener At some point, you should stop requesting location updates, possibly when o Activity loses focus o You no longer need the location

Location - Location Updates final LocationManager lm = (LocationManager) getSystemService(LOCATION_SERVICE); final LocationListener ll = new LocationListener() public void onStatusChanged(String provider, int status, Bundle extras) public void onProviderEnabled(String provider) public void onProviderDisabled(String provider) public void onLocationChanged(Location location) { Geocoder geo = new Geocoder(getApplicationContext()); List addresses = null; addresses = geo.getFromLocation(location.getLatitude(), location.getLongitude(), 1); if(addresses != null) tv.setText(addresses.get(0).getAddressLine(0)); lm.removeUpdates(this); } };

Location - Location Updates final LocationManager lm = (LocationManager) getSystemService(LOCATION_SERVICE); final LocationListener ll = new LocationListener() public void onStatusChanged(String provider, int status, Bundle extras) public void onProviderEnabled(String provider) public void onProviderDisabled(String provider) public void onLocationChanged(Location location) { Geocoder geo = new Geocoder(getApplicationContext()); List addresses = null; addresses = geo.getFromLocation(location.getLatitude(), location.getLongitude(), 1); if(addresses != null) tv.setText(addresses.get(0).getAddressLine(0)); lm.removeUpdates(this); } }; We need to register a LocationListener if we want to get updates on the location periodically

Location - Location Updates final LocationManager lm = (LocationManager) getSystemService(LOCATION_SERVICE); final LocationListener ll = new LocationListener() public void onStatusChanged(String provider, int status, Bundle extras) public void onProviderEnabled(String provider) public void onProviderDisabled(String provider) public void onLocationChanged(Location location) { Geocoder geo = new Geocoder(getApplicationContext()); List addresses = null; addresses = geo.getFromLocation(location.getLatitude(), location.getLongitude(), 1); if(addresses != null) tv.setText(addresses.get(0).getAddressLine(0)); lm.removeUpdates(this); } }; We're forced to override these methods, although we don't use them

Location - Location Updates final LocationManager lm = (LocationManager) getSystemService(LOCATION_SERVICE); final LocationListener ll = new LocationListener() public void onStatusChanged(String provider, int status, Bundle extras) public void onProviderEnabled(String provider) public void onProviderDisabled(String provider) public void onLocationChanged(Location location) { Geocoder geo = new Geocoder(getApplicationContext()); List addresses = null; addresses = geo.getFromLocation(location.getLatitude(), location.getLongitude(), 1); if(addresses != null) tv.setText(addresses.get(0).getAddressLine(0)); lm.removeUpdates(this); } }; In this callback method is where you code your response to a location change

Location - Location Updates final LocationManager lm = (LocationManager) getSystemService(LOCATION_SERVICE); final LocationListener ll = new LocationListener() public void onStatusChanged(String provider, int status, Bundle extras) public void onProviderEnabled(String provider) public void onProviderDisabled(String provider) public void onLocationChanged(Location location) { Geocoder geo = new Geocoder(getApplicationContext()); List addresses = null; addresses = geo.getFromLocation(location.getLatitude(), location.getLongitude(), 1); if(addresses != null) tv.setText(addresses.get(0).getAddressLine(0)); lm.removeUpdates(this); } }; Nothing new here...

Location - Location Updates final LocationManager lm = (LocationManager) getSystemService(LOCATION_SERVICE); final LocationListener ll = new LocationListener() public void onStatusChanged(String provider, int status, Bundle extras) public void onProviderEnabled(String provider) public void onProviderDisabled(String provider) public void onLocationChanged(Location location) { Geocoder geo = new Geocoder(getApplicationContext()); List addresses = null; addresses = geo.getFromLocation(location.getLatitude(), location.getLongitude(), 1); if(addresses != null) tv.setText(addresses.get(0).getAddressLine(0)); lm.removeUpdates(this); } }; Let's save battery life and stop listening for updates, although you may choose to stop listening for updates at some other point if you want to

Location - Location Updates See LocationManagerExample.tar

Location - Permissions In order to receive location updates from NETWORK_PROVIDER or GPS_PROVIDER, you must request user permission by declaring one or both of the following permissions o ACCESS_COARSE_LOCATION  (Wifi location) o ACCESS_FINE_LOCATION  (GPS location)

Location - Mock Location You can test your location-based features by mocking location data in the Emulator -user-location.html#MockData The Emulator uses the GPS provider Found under the Emulator Control View in DDMS, under the Telephony View

Accellerometer (Motion Sensor) Gyroscope (Motion Sensor) Light (Environment Sensor) Proximity: how close the face of a device is to an object Pressure and more See SensorListExample.tar Sensor

SensorManager lets you access the device's sensors. Get an instance of this class by calling Context.getSystemService(SENSOR_SERVICE). Always make sure to disable sensors you don't need, especially when your activity is paused o failing to do so can drain the battery in just a few hours o the system will not disable sensors automatically when the screen turns off. SensorManager

The values for each sensor may change at some point Set up an Event Listener to take action when these change These values can be retrieved from a float values[] array, regardless of the type of Sensor SensorEvent

public class SensorExampleActivity extends Activity implements SensorEventListener { SensorManager mSensorManager; Sensor mProximity; public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); mSensorManager = (SensorManager)getSystemService(SENSOR_SERVICE); mProximity = mSensorManager.getDefaultSensor(Sensor.TYPE_PROXIMITY); }... } SensorEvent

public class SensorExampleActivity extends Activity implements SensorEventListener { SensorManager mSensorManager; Sensor mProximity; public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); mSensorManager = (SensorManager)getSystemService(SENSOR_SERVICE); mProximity = mSensorManager.getDefaultSensor(Sensor.TYPE_PROXIMITY); }... } SensorEvent Because we want to know when there's a SensorEvent, when values of the Sensor change

public class SensorExampleActivity extends Activity implements SensorEventListener { SensorManager mSensorManager; Sensor mProximity; public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); mSensorManager = (SensorManager)getSystemService(SENSOR_SERVICE); mProximity = mSensorManager.getDefaultSensor(Sensor.TYPE_PROXIMITY); }... } SensorEvent Similar to how we got the Location service

public class SensorExampleActivity extends Activity implements SensorEventListener { SensorManager mSensorManager; Sensor mProximity; public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); mSensorManager = (SensorManager)getSystemService(SENSOR_SERVICE); mProximity = mSensorManager.getDefaultSensor(Sensor.TYPE_PROXIMITY); }... } SensorEvent Let's get a Sensor

public class SensorExampleActivity extends Activity implements SensorEventListener { SensorManager mSensorManager; Sensor mProximity; public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); mSensorManager = (SensorManager)getSystemService(SENSOR_SERVICE); mProximity = mSensorManager.getDefaultSensor(Sensor.TYPE_PROXIMITY); }... } SensorEvent The Proximity Sensor specifically

public class SensorExampleActivity extends Activity implements SensorEventListener { SensorManager mSensorManager; Sensor mProximity; public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); mSensorManager = (SensorManager)getSystemService(SENSOR_SERVICE); mProximity = mSensorManager.getDefaultSensor(Sensor.TYPE_PROXIMITY); }... } SensorEvent The rest of the code for this class on the next slide

public class SensorExampleActivity extends Activity implements SensorEventListener {... protected void onResume() { super.onResume(); mSensorManager.registerListener(this, mProximity, SensorManager.SENSOR_DELAY_NORMAL); } protected void onPause() { super.onPause(); mSensorManager.unregisterListener(this); } public void onSensorChanged(SensorEvent event) { Toast.makeText(this, "Proximity = " + event.values[0], Toast.LENGTH_SHORT).show(); } SensorEvent onCreate() is here somewhere, on the previous page

public class SensorExampleActivity extends Activity implements SensorEventListener {... protected void onResume() { super.onResume(); mSensorManager.registerListener(this, mProximity, SensorManager.SENSOR_DELAY_NORMAL); } protected void onPause() { super.onPause(); mSensorManager.unregisterListener(this); } public void onSensorChanged(SensorEvent event) { Toast.makeText(this, "Proximity = " + event.values[0], Toast.LENGTH_SHORT).show(); } SensorEvent Note that this is onResume()

public class SensorExampleActivity extends Activity implements SensorEventListener {... protected void onResume() { super.onResume(); mSensorManager.registerListener(this, mProximity, SensorManager.SENSOR_DELAY_NORMAL); } protected void onPause() { super.onPause(); mSensorManager.unregisterListener(this); } public void onSensorChanged(SensorEvent event) { Toast.makeText(this, "Proximity = " + event.values[0], Toast.LENGTH_SHORT).show(); } SensorEvent We only want to listen when the Activity is visible in this case, so we do this in onResume()

public class SensorExampleActivity extends Activity implements SensorEventListener {... protected void onResume() { super.onResume(); mSensorManager.registerListener(this, mProximity, SensorManager.SENSOR_DELAY_NORMAL); } protected void onPause() { super.onPause(); mSensorManager.unregisterListener(this); } public void onSensorChanged(SensorEvent event) { Toast.makeText(this, "Proximity = " + event.values[0], Toast.LENGTH_SHORT).show(); } SensorEvent We pass a Context...

public class SensorExampleActivity extends Activity implements SensorEventListener {... protected void onResume() { super.onResume(); mSensorManager.registerListener(this, mProximity, SensorManager.SENSOR_DELAY_NORMAL); } protected void onPause() { super.onPause(); mSensorManager.unregisterListener(this); } public void onSensorChanged(SensorEvent event) { Toast.makeText(this, "Proximity = " + event.values[0], Toast.LENGTH_SHORT).show(); } SensorEvent a Sensor...

public class SensorExampleActivity extends Activity implements SensorEventListener {... protected void onResume() { super.onResume(); mSensorManager.registerListener(this, mProximity, SensorManager.SENSOR_DELAY_NORMAL); } protected void onPause() { super.onPause(); mSensorManager.unregisterListener(this); } public void onSensorChanged(SensorEvent event) { Toast.makeText(this, "Proximity = " + event.values[0], Toast.LENGTH_SHORT).show(); } SensorEvent and the rate we want sensor events to be delivered

public class SensorExampleActivity extends Activity implements SensorEventListener {... protected void onResume() { super.onResume(); mSensorManager.registerListener(this, mProximity, SensorManager.SENSOR_DELAY_NORMAL); } protected void onPause() { super.onPause(); mSensorManager.unregisterListener(this); } public void onSensorChanged(SensorEvent event) { Toast.makeText(this, "Proximity = " + event.values[0], Toast.LENGTH_SHORT).show(); } SensorEvent in onPause(), we want to stop listening for updates, preserving battery life

public class SensorExampleActivity extends Activity implements SensorEventListener {... protected void onResume() { super.onResume(); mSensorManager.registerListener(this, mProximity, SensorManager.SENSOR_DELAY_NORMAL); } protected void onPause() { super.onPause(); mSensorManager.unregisterListener(this); } public void onSensorChanged(SensorEvent event) { Toast.makeText(this, "Proximity = " + event.values[0], Toast.LENGTH_SHORT).show(); } SensorEvent This is where we take action when there is a change in Sensor values

public class SensorExampleActivity extends Activity implements SensorEventListener {... protected void onResume() { super.onResume(); mSensorManager.registerListener(this, mProximity, SensorManager.SENSOR_DELAY_NORMAL); } protected void onPause() { super.onPause(); mSensorManager.unregisterListener(this); } public void onSensorChanged(SensorEvent event) { Toast.makeText(this, "Proximity = " + event.values[0], Toast.LENGTH_SHORT).show(); } SensorEvent event.values[] array has important data about the sensor event

public class SensorExampleActivity extends Activity implements SensorEventListener {... protected void onResume() { super.onResume(); mSensorManager.registerListener(this, mProximity, SensorManager.SENSOR_DELAY_NORMAL); } protected void onPause() { super.onPause(); mSensorManager.unregisterListener(this); } public void onSensorChanged(SensorEvent event) { Toast.makeText(this, "Proximity = " + event.values[0], Toast.LENGTH_SHORT).show(); } SensorEvent Refer to this page for details for each type of Sensorthis page

ProximitySensorExample.tar LightSensorExample.tar The code change between these two examples is very small SensorEvent

Using Sensors to Create a NUI NUI = Natural User Interface Human Computer Interaction is moving further and further away from the mouse and keyboard. o Being replaced by gestures on the device, air gestures, speech reco, and kinetics. There are typically two phases to gesture recognition o Data Gathering o Gesture Detection

Touch Using Gesture Detection you can register to listen for touch events on the device. o Long Press o Double Tap o Fling o Pinch o Multiple Finger gestures

Telepathy Can think of this gesture as an air gesture o Passing your hand over the device without touching the screen Might think that this would require a front facing camera to process this gesture.. o This would be an option but there is an easier way Use the Light Sensor to recognize the gesture o and the Proximity Sensor to validate

Microphone Using Speech is a no brainer for NUI o ch/RecognizerIntent.html ch/RecognizerIntent.html A more technical way of using the microphone requires signal processing o Way, way, way outside the scope of this class

Making Sense of the Data Thresholds o check peaks o filter peaks too close o based on count Time o combine with sensor data to get heart rate, for example Statistics are your friend o Since sensors spit out messy and noisy data o Mean, median, mode, range, etc

Starting Activities & Getting Results If you created two Activities A and B, you can start B from A, and have B return some result to A instead of o startActivity(Intent) call o startActivityForResult(Intent, int)

Starting Activities & Getting Results private int MY_CODE = public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); } public void startSecondActivity(View v) { Intent intent = new Intent(A.this, B.class); startActivityForResult(intent, MY_CODE); }

Starting Activities & Getting Results private int MY_CODE = public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); } public void startSecondActivity(View v) { Intent intent = new Intent(A.this, B.class); startActivityForResult(intent, MY_CODE); } This code is for Activity A

Starting Activities & Getting Results private int MY_CODE = public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); } public void startSecondActivity(View v) { Intent intent = new Intent(A.this, B.class); startActivityForResult(intent, MY_CODE); } Some number that you want to use to identify your request. #29 is nothing special here.

Starting Activities & Getting Results private int MY_CODE = public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); } public void startSecondActivity(View v) { Intent intent = new Intent(A.this, B.class); startActivityForResult(intent, MY_CODE); } I added a button to the XML and set the android:onClick attribute to this method

Starting Activities & Getting Results private int MY_CODE = public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); } public void startSecondActivity(View v) { Intent intent = new Intent(A.this, B.class); startActivityForResult(intent, MY_CODE); } We have seen this before

Starting Activities & Getting Results private int MY_CODE = public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); } public void startSecondActivity(View v) { Intent intent = new Intent(A.this, B.class); startActivityForResult(intent, MY_CODE); } Instead of startActivity(intent), we call startActivityForResult and give it the intent along with our "special" request code.

Starting Activities & Getting Results protected void onActivityResult(int requestCode, int resultCode, Intent data) { if(requestCode == MY_CODE) { if(resultCode == RESULT_OK) { // result is OK, add code here } We also need to add this method to Activity A, to react to when the result has been returned from Activity B.

Starting Activities & Getting Results public class B extends Activity public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.second); } public void finishMe(View v) { setResult(Activity.RESULT_OK, null); finish(); }

Starting Activities & Getting Results public class B extends Activity public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.second); } public void finishMe(View v) { setResult(Activity.RESULT_OK, null); finish(); } This code is for Activity B

Starting Activities & Getting Results public class B extends Activity public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.second); } public void finishMe(View v) { setResult(Activity.RESULT_OK, null); finish(); } Assuming there's a Button with android:onClick="finishMe"

Starting Activities & Getting Results public class B extends Activity public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.second); } public void finishMe(View v) { setResult(Activity.RESULT_OK, null); finish(); } Set the result of this Activity to OK. The second argument is an Intent, but we'll go in to this another time.

Starting Activities & Getting Results public class B extends Activity public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.second); } public void finishMe(View v) { setResult(Activity.RESULT_OK, null); finish(); } Finish this Activity B, which will lead to onDestroy() being called, and Activity A becoming active. onActivityResult() in A will then be called.

Starting Activities & Getting Results See StartActivityForResultExample.tar

Starting Activities & Getting Results As another example, let's call upon an existing Android Activity for a result

Starting Activities & Getting Results static final int PICK_REQUEST = public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); } public void pickContact(View v) { Intent intent = new Intent(Intent.ACTION_PICK, Phone.CONTENT_URI); startActivityForResult(intent, PICK_REQUEST); } protected void onActivityResult(int requestCode, int resultCode, Intent data) { if (requestCode == PICK_REQUEST) { if (resultCode == RESULT_OK) { /* result is OK! */ } Here's our request code again

Starting Activities & Getting Results static final int PICK_REQUEST = public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); } public void pickContact(View v) { Intent intent = new Intent(Intent.ACTION_PICK, Phone.CONTENT_URI); startActivityForResult(intent, PICK_REQUEST); } protected void onActivityResult(int requestCode, int resultCode, Intent data) { if (requestCode == PICK_REQUEST) { if (resultCode == RESULT_OK) { /* result is OK! */ } Tied to a Button with android:onClick="pickContact"

Starting Activities & Getting Results static final int PICK_REQUEST = public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); } public void pickContact(View v) { Intent intent = new Intent(Intent.ACTION_PICK, Phone.CONTENT_URI); startActivityForResult(intent, PICK_REQUEST); } protected void onActivityResult(int requestCode, int resultCode, Intent data) { if (requestCode == PICK_REQUEST) { if (resultCode == RESULT_OK) { /* result is OK! */ } We use a different constructor for the Intent this time

Starting Activities & Getting Results static final int PICK_REQUEST = public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); } public void pickContact(View v) { Intent intent = new Intent(Intent.ACTION_PICK, Phone.CONTENT_URI); startActivityForResult(intent, PICK_REQUEST); } protected void onActivityResult(int requestCode, int resultCode, Intent data) { if (requestCode == PICK_REQUEST) { if (resultCode == RESULT_OK) { /* result is OK! */ } These two arguments must mean something together, they don't always do!

Starting Activities & Getting Results static final int PICK_REQUEST = public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); } public void pickContact(View v) { Intent intent = new Intent(Intent.ACTION_PICK, Phone.CONTENT_URI); startActivityForResult(intent, PICK_REQUEST); } protected void onActivityResult(int requestCode, int resultCode, Intent data) { if (requestCode == PICK_REQUEST) { if (resultCode == RESULT_OK) { /* result is OK! */ } There should be an Activity that recognizes this Intent

Starting Activities & Getting Results static final int PICK_REQUEST = public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); } public void pickContact(View v) { Intent intent = new Intent(Intent.ACTION_PICK, Phone.CONTENT_URI); startActivityForResult(intent, PICK_REQUEST); } protected void onActivityResult(int requestCode, int resultCode, Intent data) { if (requestCode == PICK_REQUEST) { if (resultCode == RESULT_OK) { /* result is OK! */ } If none exists, you will probably get Force Close

Starting Activities & Getting Results static final int PICK_REQUEST = public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); } public void pickContact(View v) { Intent intent = new Intent(Intent.ACTION_PICK, Phone.CONTENT_URI); startActivityForResult(intent, PICK_REQUEST); } protected void onActivityResult(int requestCode, int resultCode, Intent data) { if (requestCode == PICK_REQUEST) { if (resultCode == RESULT_OK) { /* result is OK! */ } Nothing new here

Starting Activities & Getting Results static final int PICK_REQUEST = public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); } public void pickContact(View v) { Intent intent = new Intent(Intent.ACTION_PICK, Phone.CONTENT_URI); startActivityForResult(intent, PICK_REQUEST); } protected void onActivityResult(int requestCode, int resultCode, Intent data) { if (requestCode == PICK_REQUEST) { if (resultCode == RESULT_OK) { /* result is OK! */ } That starts Activity B. When Activity B has returned, the result will be returned to this callback function

Understanding Implicit Intents Implicit Intents —specify the action that should be performed and optionally the URI which should be used in conjunction with the action Intent intent = new Intent(Intent.ACTION_VIEW, Uri.parse(“ o The Android system searches for all the components that are registered to handle this specific action o If the system only finds one component, then it will launch that automatically o If the system finds more than one component, it will pop up a dialog that will give the user the option of selecting the one they prefer o If none is found, you may get an Exception.

Understanding Implicit Intents Implicit Intents How do you know which Intent Action to specify for a given Uri?

Understanding Implicit Intents There are three rules, all of which must be true for a given activity to be eligible for a given intent 1.The activity must support the specified action 2.The activity must support the stated MIME type (if supplied) 3.The activity must support all of the categories named in the intent The upshot is that you want to make your intents specific enough to find the right receiver(s), and no more specific than that.

Understanding Implicit Intents Let's take a look at what happens when when the Home key is pressed...

Understanding Implicit Intents Open LogCat Click the + button to add a new filter Enter the following o Filter Name: ActivityManager o Log Tag: ActivityManager Now press the Home key on your device o For the Tag Column, look for ActivityManager o For the Text Column, look for anything beginning with "Starting: Intent..."  The last one should be the Intent that was used to launch the home screen o Mouseover that row, and you should see values for the following  act, which is the action - android.intent.action.MAIN  cat, which is the category - android.intent.category.HOME  in this case, there is no data, but sometimes there is o Note the action and category

Understanding Implicit Intents Can we create our own Home Screen app? Create a new project Open AndroidManifest.xml o add a new intent-filter to your Activity o the action should match the one you found in LogCat o so should the category o in addition, add the DEFAULT category

Understanding Implicit Intents Now press the Home key... For a nicer effect, make your Activity full-screen by adding this to the tag in the manifest file

Understanding Implicit Intents See HomeScreenExample.tar

Understanding Implicit Intents It's not too difficult to figure out the Java code for the same Intent: or IntentFilter myFilter = new IntentFilter(Intent.ACTION_MAIN); myFilter.addCategory(Intent.CATEGORY_HOME);

Understanding Implicit Intents If you make your Intent Filters too vague too specific it won't match what you expect it to. Try to make your Intent Filters precise!

References The Busy Coder's Guide to Android Development - Mark Murphy The Busy Coder's Guide to Android Development - Mark Murphy Android Developers The Mobile Lab at Florida State University