Saturday, November 24, 2012

Silence And More!!

“Time:11.30am-12.30pm Software Release Plan Meeting” popped up from my calendar. Before I get into the meeting I answer a call to make a quick conversation with my colleague. In the middle of the meeting my phone buzzes out again. Stumbling with my phone I give an apologetic look to all the gentlemen in the room and obediently put my mobile on "Silent Mode".

The meeting gets over in an hour and I get into discussion with my team on some of the key issues. Lunch time I realize that my phone was muted way long back. I quickly un-mute my phone and in the process come across couple of missed calls.

In this blog I introduce an application which helps me to
  • Mute the phone for the period that has been set
  • Automatically un-mutes the phone once the period has expired
When the phone was set in silent mode all the incoming calls are muted. Other key features supported during this silence period are
  • Automatic SMS notification to your phone contacts when they try to reach you. This means SMS notification is not send out to numbers that are not listed in your contacts.
  • Duplicate SMS notifications are suppressed if the same contact tries to reach you again.
  • Displays an alert in the mobile's notification tray if any of your phone contact tried to reach you.
  • Cancel the silence period at any point of time and the phone gets un-muted.
  • Track the calls that are listed in my phone and store them as part of its history

Below is a sample snapshot and the details shown by this application. 


    
  Following are the details one can see:
  • Notification icon indicating that a contact had made a call when the call was in silent mode.
  • Notification icon indicating that the phone is in mute.
  • Placeholder for setting the silence interval.
  • History indicating the period of silence mode.
  • And list of contacts who have called in this interval.
    
Let's briefly walk over different frameworks and API that I used to build this application. 

  •  Presentation Layer
  • Being a part time web developer, the user interface is made of html, css and javascripts. All my presentation logic are composed of these three main UI elements. This includes deriving the time, rendering history data to the "li" elements of html. When the user selects the interval and submits, a javascript call is made into the phone gap plugin which is discussed below.  
  • Cordova or PhoneGap Plugins
  • The presentation layer uses the Cordova or PhoneGap plugin to interact with the android platform. The javascript calls from the presentation layer uses the Cordova plug-in to interact with the android platform related operations. Since there are numerous examples in the internet showing how this plugin can be installed, configured into your eclipse IDE and used with android platform I will not be repeating this.
    Below is a quick view on how Cordova is used and how the plugin can be defined.
    import org.apache.cordova.DroidGap;
    public class myActivity extends DroidGap {
       @Override
        public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
            //do all activity logic...
            //invoke the presentation layer
            super.loadUrl("file:///android_asset/www/index.html");
         }
    }
    import org.apache.cordova.api.Plugin;
    import org.apache.cordova.api.PluginResult;


    public class silenceAndMore extends Plugin { 
       @Override 
       public PluginResult execute(String action, JSONArray data, String callbackId) { 
          //all actions from presentation layer to platform are written here
         return new PluginResult(PluginResult.Status.OK, "Success!!");
       }
    }
  • Alarm Manager
  • Provides an inbuilt functionality of “timers” in android. This is basically used to set the period of silence and to wake up when the period has expired. The main reason for choosing the Alarm Manager is because of its ability to run in the background even when the main activity is removed by the android platform. Below example shows how typically an interval can be set in Alarm Manager 
    Calendar cal = Calendar.getInstance();             
    // Set the calendar with your date and time
    cal.set(Calendar.DAY_OF_MONTH, date);
    cal.set(Calendar.HOUR_OF_DAY, hour);
    cal.set(Calendar.MINUTE, min);
    //create an intent for the alarm
    Intent alarmIntent = new Intent(context, AlarmReceiver.class);
    //set an action in the intent
    alarmIntent.setAction("ALARM");         alarmIntent.addFlags(Intent.FLAG_ACTIVITY_NEWTASK);
    //cancel all old and pending intents
    PendingIntent pendingIntent = PendingIntent.getBroadcast(context, 0,   alarmIntent, PendingIntent.FLAG_CANCEL_CURRENT);
    //Get the alarm manager and set the time
    AlarmManager alarmManager = (AlarmManager)getSystemService(Context.ALARM_SERVICE);
    alarmManager.set(AlarmManager.RTC_WAKEUP, cal.getTimeInMillis(), pendingIntent);

  • Broadcast Receiver
  • Android’s broadcast receiver receives all the callbacks that are registered. The callbacks for alarms and calls are registered respectively. When the timer has expired the system wakes up the broadcast receiver. Similarly the incoming calls with various phone states are triggered for call registrations.
    public class myReceiver extends BroadcastReceiver {
      String PHONESTATE="android.intent.action.PHONE_STATE";
      @Override
      public void onReceive(Context context, Intent intent) {
        //Switch the usage based on action on the intent 
        if (intent.getAction().equals("ALARM")){    
           //do logic for expiry of interval
        }else if (intent.getAction().equals(PHONESTATE)){    
             //check the phone states using Telephony manager
             //i.e if it is idle or ringing
        }
      }
    }
  • Telephony Manager
  • This "Phone State" intent is thrown to broadcast receiver for an incoming call. This intent will have calling party number and at the same time the state of the call. The telephony manager helps to validate  the state of the call. An example shows below how an incoming call can be detected.

    if(intent.getAction().equals(PHONESTATE)) {   
    //get the phone state
    String state= intent.getStringExtra(TelephonyManager.EXTRA_STATE);
    //get the phone number
    String number= intent.getStringExtra(TelephonyManager.EXTRA_INCOMING_NUMBER);
    if (TelephonyManager.EXTRA_STATE_RINGING.equals(state)){
        //call state is ringing. Do your logic here.
    }


  • Audio Manager 
  • Helps in setting the phone in silent or remove from silence mode. When the user sets the interval, phone is moved into silent mode. And when the timer is expired, the phone is removed from the silent mode. Below example shows how a phone can be set to a "Silent" mode.

    AudioManager mAM = getSystemService(context.AUDIO_SERVICE);
    mAM.setRingerMode(AudioManager.RINGER_MODE_SILENT);

  • Contacts Provider
  • The contacts provider class lets you access the contacts in the phone. Along with the content resolver one can access the phone contact book and access the "Name" from the incoming number. Below is a typical query into the phone contact book to get the display name from a phone number.
    ContentResolver localContentResolver = context.getContentResolver();
    Cursor contactLookupCursor =  localContentResolver.query(
                Uri.withAppendedPath(PhoneLookup.CONTENT_FILTER_URI, 
                Uri.encode(number)), 
                new String[]{PhoneLookup.DISPLAY_NAME},                             
                null, null , null);                  
  • Notification Manager
  • This class helps to set a notification icon on the phone’s notification tray. When the incoming call is from one of the phone contacts, the notification tray is enabled with the application's icon. The sample for this notification manager is present in the android development site. I followed most of it to set the notification.
  • SMS Manager 
  • To send SMS text message to a recipient. The application after retrieving the contact name for the incoming number will send SMS to this number.
    //set the SMS text message to be send
    String payload = "Sorry I am busy right now";
    //Get the SMS Manager instance
    SmsManager smsManager = SmsManager.getDefault();
    //send the text to the number
    smsManager.sendTextMessage(number, null, payload, null, null);
  • File Storage
  • The information like time of last silence period, contacts who called are persisted in a file. The data was persisted using the basic FileInputstream and FileOutputstream. A typical file read operation using an FileInputStream is shown below.
      FileInputStream fIn = null;
      StringBuffer strContent = new StringBuffer("");
      int ch;
      try {
            //open the file under this context
            fIn = openFileInput("data.txt");
            while ((ch = fIn.read()) != -1)
                  strContent.append((char) ch);
            fIn.close();
          } catch (Exception e) {
            System.out.println(e);
            return null;
          }
      String data = strContent.toString();


    Conclusion

    My earlier design was making use of javascripts for functions like setting intervals, maintaining history etc. Additionally I used the Cordova plugin to retrieve the contact names from the phone number. But when the application goes into background, the android platform removes the old application and relinquishes its memory to new applications that come into the foreground. So when you set a silence interval of say an hour or two its very likely that the application gets removed. Now that the silence mode was set the phone would never un-mute as the application was removed by android platform.   

    With this I conclude the session of describing the use-cases and frameworks for this application that I have designed and implemented. I will be very glad to present any more free sample for this application or answer any queries that you have here.