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;

网页地址:

https://www.androidos.net.cn/android/8.0.0_r4/xref/packages/apps/Messaging/src/com/android/messaging/ui/conversationlist/ShareIntentActivity.java

找到之后,在网页中查找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_FAILURESentIntent可能包含无线电技术特定值的额外“错误代码”,通常只对故障排除有用。
基于每个应用程序的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=========================

posted @ 2018-08-15 21:49  绿叶萌飞  阅读(344)  评论(0编辑  收藏  举报