目录
短信发送器demo地址:https://github.com/liuchenyang0515/SmsSend
模拟效果:
我们初步想要的功能就是ListView显示一些数据,点击这些数据后想要发送一些祝福短信,要求把这些内容直接显示在发送短信界面的编辑框内,方便发送,因为这个不是我们自己写的界面,所以用隐式意图。为了查找putExtra的键,我们需要查看系统源码。
MainActivity.java
import android.content.Intent; import android.os.Bundle; import android.support.v7.app.AppCompatActivity; import android.view.View; import android.widget.AdapterView; import android.widget.ArrayAdapter; import android.widget.ListView; public class MainActivity extends AppCompatActivity { String[] s = {"真心的编织快乐,细心的装满幸福,小心的盛上吉祥,用心的放入喜庆,耐心的把它们进行包装,诚心的用短信发送给你...", "周末又已来到,祝福轮番开炮,瞄准劳累之敌,把疲惫郁闷轰掉,打响开心号弹,吹起愉悦号角,冲向快乐山头,让舒畅飘扬高高...", "人生忙忙碌碌,日子酸酸甜甜,缘分简简单单,联系断断续续,惦记时时刻刻,祝福长长久久,天天开开心心,祝你平平安安,周末愉快!", "勤勤工作,兢兢业业报国志。肯肯上进,升职加薪见几人?东奔西跑,忙忙碌碌周身痛..."}; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); // 找到lv ListView lv = (ListView) findViewById(R.id.lv); // 设置数据 ArrayAdapter<String> adapter = new ArrayAdapter<>(this, R.layout.item, s); // 设置数据适配器 lv.setAdapter(adapter); // 给listview设置点击事件 lv.setOnItemClickListener(new AdapterView.OnItemClickListener() { @Override public void onItemClick(AdapterView<?> parent, View view, int position, long id) { // 把点击条目数据取出来,数据在哪里存着就去哪里取 String content = s[position]; // 跳转到发送短信的页面 // 查询源码得到意图过滤器 // 源码查看地址https://www.androidos.net.cn/android/8.0.0_r4/xref/packages/apps/Messaging/AndroidManifest.xml /*<intent-filter android:label="@string/share_intent_label"> <action android:name="android.intent.action.SEND" /> <category android:name="android.intent.category.DEFAULT" /> <data android:mimeType="text/plain" /> <data android:mimeType="text/x-vCard" /> <data android:mimeType="text/x-vcard" /> <data android:mimeType="image/*" /> <data android:mimeType="audio/*" /> <data android:mimeType="application/ogg" /> </intent-filter>*/ Intent intent = new Intent(); // 设置action intent.setAction("android.intent.action.SEND"); // 添加category intent.addCategory("android.intent.category.DEFAULT"); // 设置type intent.setType("text/plain"); // 传递数据 intent.putExtra(Intent.EXTRA_TEXT, content); // 跳转到发送短信的页面 startActivity(intent); } }); } }
查找到android系统源码目录:packages/apps/Messaging/AndroidManifest.xml
地址:
https://www.androidos.net.cn/android/8.0.0_r4/xref/packages/apps/Messaging/AndroidManifest.xml
查找动作android.intent.action.SEND去匹配,发现如下:
<!-- Handles sharing intent --> <activity android:name=".ui.conversationlist.ShareIntentActivity" android:configChanges="orientation|screenSize|keyboardHidden" android:screenOrientation="user" android:theme="@style/BugleTheme.DialogActivity" android:excludeFromRecents="true" android:documentLaunchMode="always"> <intent-filter android:label="@string/share_intent_label"> <action android:name="android.intent.action.SEND" /> <category android:name="android.intent.category.DEFAULT" /> <data android:mimeType="text/plain" /> <data android:mimeType="text/x-vCard" /> <data android:mimeType="text/x-vcard" /> <data android:mimeType="image/*" /> <data android:mimeType="audio/*" /> <data android:mimeType="application/ogg" /> </intent-filter> <intent-filter android:label="@string/share_intent_label"> <action android:name="android.intent.action.SEND_MULTIPLE" /> <category android:name="android.intent.category.DEFAULT" /> <data android:mimeType="image/*" /> </intent-filter> </activity>
看到这个动作是ui.conversationlist.ShareIntentActivity里面的,然后去这个里面找接收数据的键来确定我们发送数据的键是什么。
是哪个包下面的呢?
<manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.android.messaging" android:installLocation="internalOnly">
原来是com.android.messaging包底下的
即package com.android.messaging.ui.conversationlist;
网页地址:
找到之后,在网页中查找getStringExtra去匹配文本
发现了一个Intent.EXTRA_TEXT,这就是我们想要的键
然后putExtra进去的就是这个键。
activity_main.xml
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical"> <ListView android:id="@+id/lv" android:layout_width="match_parent" android:layout_height="match_parent"></ListView> </LinearLayout>
item.xml
<?xml version="1.0" encoding="utf-8"?> <TextView xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="wrap_content" />
运行结果(真机测试):
点击一个item,分享到“信息”
然后发现item里面的内容已经在编辑框内了。
短信发送器:
MainActivity.java
import android.Manifest; import android.app.Activity; import android.app.AlertDialog; import android.content.DialogInterface; import android.content.Intent; import android.content.pm.PackageManager; import android.support.annotation.NonNull; import android.support.v4.app.ActivityCompat; import android.support.v4.content.ContextCompat; import android.support.v7.app.AppCompatActivity; import android.os.Bundle; import android.telephony.SmsManager; import android.view.View; import android.widget.EditText; import android.widget.Toast; import java.util.List; public class MainActivity extends AppCompatActivity { private EditText et_number; private EditText et_content; private static final int REQUESTCODE_ADD = 1; private static final int REQUESTCODE_INSERT = 2; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); // 找到控件 et_number = (EditText) findViewById(R.id.et_number); et_content = (EditText) findViewById(R.id.et_content); } public void onclick(View view) { switch (view.getId()) { case R.id.btn_add: Intent intent = new Intent(this, ContactActivity.class); startActivityForResult(intent, REQUESTCODE_ADD); break; case R.id.btn_send: // 获取动态权限 if (ContextCompat.checkSelfPermission(MainActivity.this, Manifest.permission.SEND_SMS) != PackageManager.PERMISSION_GRANTED) { ActivityCompat.requestPermissions(MainActivity.this, new String[]{Manifest.permission.SEND_SMS}, 1); } else { sendSms(); } break; case R.id.btn_insert: Intent intent2 = new Intent(this, SmsTemplateActivity.class); startActivityForResult(intent2, REQUESTCODE_INSERT); break; } } private void sendSms() { // 获取发送短信的号码和发送的内容 String number = et_number.getText().toString().trim(); String content = et_content.getText().toString().trim(); // 获取SmsManager实例 SmsManager smsManager = SmsManager.getDefault(); List<String> divideMessage = smsManager.divideMessage(content); for (String div : divideMessage) { smsManager.sendTextMessage(number, null, div, null, null); } } @Override public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { switch (requestCode) { case 1: if (grantResults.length > 0) { for (int result : grantResults) { if (result != PackageManager.PERMISSION_GRANTED) { if (ActivityCompat.shouldShowRequestPermissionRationale(this, Manifest.permission.SEND_SMS)) { showPermissionDialog(permissions); } else { Toast.makeText(this, "您已拒绝权限,请在设置手动打开权限", Toast.LENGTH_SHORT).show(); } return; } } sendSms(); } break; } } private void showPermissionDialog(final String[] permissions) { AlertDialog.Builder dialog = new AlertDialog.Builder(this); dialog.setTitle("提示!"); dialog.setMessage("这个权限关系到发送短信,如拒绝需要在设置手动打开!"); dialog.setCancelable(false); dialog.setPositiveButton("授权", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { ActivityCompat.requestPermissions(MainActivity.this, permissions, 1); } }); dialog.setNegativeButton("拒绝", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { } }); dialog.show(); } // 当我们开启的activity页面关闭的时候调用此方法 @Override protected void onActivityResult(int requestCode, int resultCode, Intent data) { switch (requestCode) { case REQUESTCODE_ADD: if (resultCode == RESULT_OK) { String phone = data.getStringExtra("phone"); et_number.setText(phone); } break; case REQUESTCODE_INSERT: if (resultCode == RESULT_OK) { String smsContent = data.getStringExtra("smsContent"); //et_content.setText(smsContent); et_content.append(smsContent); } break; } } }
批注:
public void sendTextMessage (String destinationAddress, String scAddress,
String text, PendingIntent sentIntent, PendingIntent deliveryIntent)
发送基于短信的文本。
注:使用此方法需要您的应用程序具有Manifest.permission.SEND_SMS
许可。
注:从Android4.4开始(API级别19),如果只有当不选择应用程序作为默认SMS应用程序,系统会自动将使用此方法发送的消息写入SMS提供者(默认SMS应用始终负责将其发送的消息写入SMS提供程序)。有关如何作为默认SMS应用程序运行的信息,请参阅Telephony
.
参数 | |
---|---|
destinationAddress |
String: 要发送消息的地址
|
scAddress |
String: 是服务中心地址或null以使用当前默认SMSC
|
text |
String: 要发送的消息正文
|
sentIntent |
PendingIntent: 如果不为空,则为PendingIntent 消息成功发送或失败时广播。结果代码将是Activity.RESULT_OK 对于成功,或其中的一个错误:RESULT_ERROR_GENERIC_FAILURE RESULT_ERROR_RADIO_OFF RESULT_ERROR_NULL_PDU 为 RESULT_ERROR_GENERIC_FAILURE SentIntent可能包含无线电技术特定值的额外“错误代码”,通常只对故障排除有用。基于每个应用程序的SMS控制检查哨兵。如果SentIntent为NULL,调用者将根据所有未知的应用程序进行检查,这将导致在检查期间发送较少数量的SMS。
|
deliveryIntent |
PendingIntent: 如果不为空,则为PendingIntent 消息传递到收件人时广播。状态报告的原始PDU位于扩展数据(“PDU”)中。
|
抛出 | |
---|---|
IllegalArgumentException |
如果destinationAddress或Text为空 |
短信太长就会分成几封发送出去,所以要用divideMessage
public ArrayList<String> divideMessage (String text)
将一个消息文本分成几个片段,没有一个大于最大SMS消息大小。
参数 | |
---|---|
text |
String: 原始信息。不能是空的。
|
回报 | |
---|---|
ArrayList<String> |
阿ArrayList 的字符串,按顺序组成原始消息。
|
抛出 | |
---|---|
IllegalArgumentException |
如果文本为空 |
ContactActivity.java
import android.content.Intent; import android.os.Bundle; import android.support.v7.app.AppCompatActivity; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.AdapterView; import android.widget.BaseAdapter; import android.widget.ListView; import android.widget.TextView; import java.util.ArrayList; import java.util.List; public class ContactActivity extends AppCompatActivity { private List<Person> lists; private ListView lv; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); // 加载布局 setContentView(R.layout.activity_contact); lv = (ListView) findViewById(R.id.lv); lists = new ArrayList<>(); for (int i = 0; i < 20; ++i) { Person p = new Person(); p.setName("张三"); p.setPhone("11" + i); lists.add(p); } // 展示数据 lv.setAdapter(new Myadapter()); // 给listview设置点击事件 lv.setOnItemClickListener(new AdapterView.OnItemClickListener() { // parent代表listview,view代表item,可在断点调试验证 @Override public void onItemClick(AdapterView<?> parent, View view, int position, long id) { // 获取点中item的数据 String phone = lists.get(position).getPhone(); // 把数据返回给调用者 Intent intent = new Intent(); intent.putExtra("phone", phone); // 把结果返回给调用者 setResult(RESULT_OK, intent); // 关闭当前页面 finish(); // finish后通过onActivityResult返回数据 } }); } @Override public void onBackPressed() { } private class Myadapter extends BaseAdapter { @Override public int getCount() { return lists.size(); } @Override public Object getItem(int position) { return null; } @Override public long getItemId(int position) { return 0; } @Override public View getView(int position, View convertView, ViewGroup parent) { View view; ViewHolder viewHolder; if (convertView == null) { // 获取子布局 view = LayoutInflater.from(getApplicationContext()).inflate(R.layout.contact_item, parent, false); viewHolder = new ViewHolder(); // 找到在item中定义的控件来显示数据 // 一定要写view.findViewById,findViewById是有上下文的,默认是在Activity的主布局中 viewHolder.tv_name = (TextView) view.findViewById(R.id.tv_name); viewHolder.tv_phone = (TextView) view.findViewById(R.id.tv_phone); view.setTag(viewHolder); } else { view = convertView; viewHolder = (ViewHolder) view.getTag(); } // 展示数据 viewHolder.tv_name.setText(lists.get(position).getName()); viewHolder.tv_phone.setText(lists.get(position).getPhone()); return view; } private class ViewHolder { TextView tv_name, tv_phone; } } }
Person.java
public class Person { private String name; private String phone; public String getName() { return name; } public void setName(String name) { this.name = name; } public String getPhone() { return phone; } public void setPhone(String phone) { this.phone = phone; } }
SmsTemplateActivity.java
import android.content.Intent; import android.os.Bundle; import android.support.v7.app.AppCompatActivity; import android.view.View; import android.widget.AdapterView; import android.widget.ArrayAdapter; import android.widget.ListView; public class SmsTemplateActivity extends AppCompatActivity { String[] str = {"我在吃饭,请稍后联系", "我在开会,请稍后联系", "我在上课,请稍后联系", "我在加班,请稍后联系","我在约会,请稍后联系"}; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_smstemplate); ListView lv = (ListView) findViewById(R.id.lv); ArrayAdapter<String> adapter = new ArrayAdapter<>(this, R.layout.smstemplate_item, R.id.tv, str); // 显示数据 lv.setAdapter(adapter); // 设置item点击事件 lv.setOnItemClickListener(new AdapterView.OnItemClickListener() { @Override public void onItemClick(AdapterView<?> parent, View view, int position, long id) { // 取出点击item的数据 String smsContent = str[position]; // 把smsContent返回给调用者 Intent intent = new Intent(); intent.putExtra("smsContent", smsContent); setResult(RESULT_OK, intent); finish(); } }); } }
activity_main.xml
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical"> <RelativeLayout android:layout_width="match_parent" android:layout_height="wrap_content"> <EditText android:id="@+id/et_number" android:layout_width="match_parent" android:layout_height="wrap_content" android:hint="请输入手机号码" /> <Button android:id="@+id/btn_add" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_alignBottom="@id/et_number" android:layout_marginBottom="8dp" android:layout_alignParentRight="true" android:onClick="onclick" android:text="+" /> </RelativeLayout> <EditText android:id="@+id/et_content" android:layout_width="match_parent" android:gravity="top" android:layout_height="200dp" android:hint="请输入发送短信的内容" /> <Button android:id="@+id/btn_insert" android:layout_width="wrap_content" android:layout_height="wrap_content" android:onClick="onclick" android:text="插入短信模版"/> <Button android:id="@+id/btn_send" android:layout_width="wrap_content" android:layout_height="wrap_content" android:onClick="onclick" android:text="发送"/> </LinearLayout>
activity_contact.xml
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent"> <ListView android:id="@+id/lv" android:layout_width="match_parent" android:layout_height="match_parent"></ListView> </LinearLayout>
contact_item.xml
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="wrap_content" android:orientation="horizontal"> <TextView android:id="@+id/tv_name" android:layout_width="0dp" android:layout_height="wrap_content" android:layout_weight="1" android:text="aaa" android:textSize="20sp" /> <TextView android:id="@+id/tv_phone" android:layout_width="0dp" android:layout_height="wrap_content" android:layout_weight="1" android:text="110" android:textSize="20sp" /> </LinearLayout>
activity_smstemplate.xml
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent"> <ListView android:id="@+id/lv" android:layout_width="match_parent" android:layout_height="match_parent"></ListView> </LinearLayout>
smstemplate_item.xml
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent"> <TextView android:id="@+id/tv" android:layout_width="match_parent" android:layout_height="wrap_content" android:textSize="20sp"/> </LinearLayout>
AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.example.sms_send"> <uses-permission android:name="android.permission.SEND_SMS" /> <application android:allowBackup="true" android:icon="@mipmap/ic_launcher" android:label="@string/app_name" android:roundIcon="@mipmap/ic_launcher_round" android:supportsRtl="true" android:theme="@style/AppTheme"> <activity android:name=".MainActivity"> <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity> <activity android:name=".ContactActivity" /> <activity android:name=".SmsTemplateActivity" /> </application> </manifest>
具体代码见github:https://github.com/liuchenyang0515/SmsSend
==========================Talk is cheap, show me the code=========================
来源:https://www.cnblogs.com/lcy0515/p/10807872.html