USSD в Android
USSD (Unstructured Supplementary Service Data)— стандартный сервис в сетях GSM, позволяющий организовать интерактивное взаимодействие между абонентом сети и сервисным приложением в режиме передачи коротких сообщений.
Как известно, Android не имеет API для чтения USSD сообщений, далее я расскажу как решить эту проблему.
Как то столкнулся с задачей отослать команду и принять USSD сообщение. Оказывается USSD ответ сохраняется в буфере BufferedReader и существует сторонний класс USSD, для парсинга информации из этого буфера.
Сам класс
package com.example.android.UssdMessage; import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; import java.util.Calendar; import java.util.Date; import java.text.ParseException; import java.text.SimpleDateFormat; import android.util.Log; public class USSD { private static String startmsg="displayMMIComplete"; //start msg to look for private static String endmsg="MMI code has finished running"; //stop msg private static String trimmsg="- using text from MMI message: '"; //a msg to remove from the text private long before=3000; //delay (ms) before creation of the class before a msg (USDD) is valid (use timestamp) private long after=3000; //delay (ms) after creation of the class that a msg (USDD) is valid (wait after ms) private String msg=""; //the USSD message private boolean found=false; private long t=-1; //timestamp of the found log public USSD() { this(3000,3000); } //USSD in log : example public USSD(long before_creation,long after_creation) { before=before_creation; after=after_creation; long timestamp=System.currentTimeMillis(); //creation of the class --> look for the USSD msg in the logs Log.d("USSDClass", "Class creation - timestamp: "+String.valueOf(timestamp)); try { //sample code taken from alogcat ... Process logcatProc = Runtime.getRuntime().exec("logcat -v time -b main PhoneUtils:D"); //get PhoneUtils debug log with time information BufferedReader mReader = new BufferedReader(new InputStreamReader(logcatProc.getInputStream()), 1024); String line=""; boolean tostop=false; long stop=timestamp+after; //to stop the while after "after" ms while (((line = mReader.readLine()) != null)&&(System.currentTimeMillis()<stop)&&(tostop==false)) { if (line.length()>19) //the line should be at least with a length of a timestamp (19) ! { if (line.contains(startmsg)) //check if it is a USSD msg { //log example : "12-10 20:36:39.321 D/PhoneUtils( 178): displayMMIComplete: state=COMPLETE" t=extracttimestamp(line); //extract the timestamp of thie msg Log.d("USSDClass", "Found line at timestamp : "+String.valueOf(t)); if (t>=timestamp-before) found=true; //start of an USDD is found & is recent ! } else if (found) { //log example : "12-10 20:36:39.321 D/PhoneUtils( 178): displayMMIComplete: state=COMPLETE" if (line.contains(endmsg)) tostop=true; else { //log example : "12-10 20:36:39.321 D/PhoneUtils( 178): - using text from MMI message: 'Your USSD message with one or several lines" Log.d("USSDClass", "Line content : "+line); String[] v=line.split("\\): "); //doesn't need log information --> split with "): " separator if (v.length>1) msg+=v[1].replace(trimmsg, "").trim()+"\n"; } } } } } catch (IOException e) { Log.d("USSDClass", "Exception:"+e.toString()); } } public boolean IsFound() { return found; } public String getMsg() { return msg; } //extract timestamp from a log line with format "MM-dd HH:mm:ss.ms Level/App:msg" Example : 12-10 20:36:39.321 //Note : known bug : happy new year check will not work !!! private long extracttimestamp(String line) { long timestamp=-1; //default value if no timestamp is found String[] v=line.split(" "); if (v.length>1) //check if there is space { Calendar C=Calendar.getInstance(); int y=C.get(Calendar.YEAR); String txt=v[0]+"-"+y+" "+v[1]; //transform in format "MM-dd-yyyy HH:mm:ss" SimpleDateFormat formatter = new SimpleDateFormat("MM-dd-yyyy HH:mm:ss"); try { Date tmp=formatter.parse(txt); timestamp=tmp.getTime(); String[] ms=v[1].split("."); //get ms if (ms.length>1) timestamp+=Integer.getInteger(ms[1]); } catch (ParseException e) { Log.d("USSDClass", "USDD.extractimestamp exception:"+e.toString()); } } return timestamp; } }
Теперь рассмотрим саму реализацию использования класса:
Для начала создадим интерфейс приложения, у нас будет AutoCompleteTextView, TextView, Button:
<!--?xml version="1.0" encoding="utf-8"?--> <linearlayout xmlns:android="schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="fill_parent" android:layout_height="fill_parent"> <autocompletetextview android:layout_width="fill_parent" android:text="" android:layout_height="wrap_content" android:inputtype="phone|textUri" android:id="@+id/Text1"> <requestfocus></requestfocus> </autocompletetextview> <textview android:layout_width="fill_parent" android:id="@+id/Text2" android:layout_height="wrap_content"> </textview> <button android:text="@string/send" android:id="@+id/button1" android:layout_width="fill_parent" android:layout_height="wrap_content"> </button> </linearlayout>
Вот такой интерфейс у нас должен получиться:
Теперь предадим жизни приложению, напишем сам код получения результата USSD.
// не пишу все импорты, напишу только то, что не забудьте подключить класс USSD import com.example.android.UssdMessage.USSD; public class UssdmessageActivity extends Activity implements OnClickListener { /** Called when the activity is first created. */ private TextView view; private AutoCompleteTextView number; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); Button button = (Button) findViewById(R.id.button1); button.setOnClickListener(this); this.view = (TextView) findViewById(R.id.Text2); this.number = (AutoCompleteTextView) findViewById(R.id.Text1); } @Override public void onClick(View arg0) { String encodedHash = Uri.encode("#"); call("*" + number.getText() + encodedHash); this.view.setText(""); } protected void call(String phoneNumber) { try { startActivityForResult( new Intent("android.intent.action.CALL", Uri.parse("tel:" + phoneNumber)), 1); } catch (Exception eExcept) { this.view.append("\n\n " + "\n" + eExcept.toString()); } } @Override protected void onActivityResult(int requestCode, int resultCode, Intent data) { USSD ussd = new USSD(4000,4000); // передается два параметра, задержка до и после (ms) создания сообщения if (ussd.IsFound()) this.view.append("\n"+ussd.getMsg()); else this.view.append(""+R.string.error_ussd_msg); } }
Скриншот работы приложения:
Добавлю то, что класс написан так, что выводит в logcat отладочную информацию.
На мой взгляд не сложный и удобный класс, который может помочь в решении Ваших задач.