android学习笔记----短信发送器
目录
短信发送器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=========================