zoukankan      html  css  js  c++  java
  • app开发快速理解——推送通知&小红点角标

     

    package com.example.chapter11;
    
    import android.app.Notification;
    import android.app.NotificationManager;
    import android.app.PendingIntent;
    import android.content.Context;
    import android.content.Intent;
    import android.graphics.BitmapFactory;
    import android.os.Build;
    import android.os.Bundle;
    import android.text.TextUtils;
    import android.view.View;
    import android.widget.EditText;
    import android.widget.Toast;
    
    import androidx.appcompat.app.AppCompatActivity;
    
    import com.example.chapter11.util.ViewUtil;
    
    public class NotifySimpleActivity extends AppCompatActivity implements View.OnClickListener {
        private EditText et_title;
        private EditText et_message;
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_notify_simple);
            et_title = findViewById(R.id.et_title);
            et_message = findViewById(R.id.et_message);
            findViewById(R.id.btn_send_simple).setOnClickListener(this);
        }
    
        @Override
        public void onClick(View v) {
            if (v.getId() == R.id.btn_send_simple) {
                ViewUtil.hideOneInputMethod(this, et_message); // 隐藏输入法软键盘
                if (TextUtils.isEmpty(et_title.getText())) {
                    Toast.makeText(this, "请填写消息标题", Toast.LENGTH_SHORT).show();
                    return;
                }
                if (TextUtils.isEmpty(et_message.getText())) {
                    Toast.makeText(this, "请填写消息内容", Toast.LENGTH_SHORT).show();
                    return;
                }
                String title = et_title.getText().toString();
                String message = et_message.getText().toString();
                sendSimpleNotify(title, message); // 发送简单的通知消息
            }
        }
    
        // 发送简单的通知消息(包括消息标题和消息内容)
        private void sendSimpleNotify(String title, String message) {
            // 发送消息之前要先创建通知渠道,创建代码见MainApplication.java
            // 创建一个跳转到活动页面的意图
            Intent clickIntent = new Intent(this, MainActivity.class);
            // 创建一个用于页面跳转的延迟意图
            PendingIntent contentIntent = PendingIntent.getActivity(this,
                    R.string.app_name, clickIntent, PendingIntent.FLAG_UPDATE_CURRENT);
            // 创建一个通知消息的建造器
            Notification.Builder builder = new Notification.Builder(this);
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
                // Android 8.0开始必须给每个通知分配对应的渠道
                builder = new Notification.Builder(this, getString(R.string.app_name));
            }
            builder.setContentIntent(contentIntent) // 设置内容的点击意图
                    .setAutoCancel(true) // 点击通知栏后是否自动清除该通知
                    .setSmallIcon(R.mipmap.ic_launcher) // 设置应用名称左边的小图标
                    .setSubText("这里是副本") // 设置通知栏里面的附加说明文本
                    // 设置通知栏右边的大图标
                    .setLargeIcon(BitmapFactory.decodeResource(getResources(), R.drawable.ic_app))
                    .setContentTitle(title) // 设置通知栏里面的标题文本
                    .setContentText(message); // 设置通知栏里面的内容文本
            Notification notify = builder.build(); // 根据通知建造器构建一个通知对象
            // 从系统服务中获取通知管理器
            NotificationManager notifyMgr = (NotificationManager)
                    getSystemService(Context.NOTIFICATION_SERVICE);
            // 使用通知管理器推送通知,然后在手机的通知栏就会看到该消息
            notifyMgr.notify(R.string.app_name, notify);
        }
    
    }
    

      

    package com.example.chapter11;
    
    import android.app.Notification;
    import android.app.NotificationManager;
    import android.app.PendingIntent;
    import android.content.Context;
    import android.content.Intent;
    import android.graphics.BitmapFactory;
    import android.os.Build;
    import android.os.Bundle;
    import android.text.TextUtils;
    import android.view.View;
    import android.widget.EditText;
    import android.widget.Toast;
    
    import androidx.appcompat.app.AppCompatActivity;
    
    import com.example.chapter11.util.ViewUtil;
    
    public class NotifyCounterActivity extends AppCompatActivity implements View.OnClickListener {
        private EditText et_title;
        private EditText et_message;
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_notify_counter);
            et_title = findViewById(R.id.et_title);
            et_message = findViewById(R.id.et_message);
            findViewById(R.id.btn_send_counter).setOnClickListener(this);
        }
    
        @Override
        public void onClick(View v) {
            if (v.getId() == R.id.btn_send_counter) {
                ViewUtil.hideOneInputMethod(this, et_message); // 隐藏输入法软键盘
                if (TextUtils.isEmpty(et_title.getText())) {
                    Toast.makeText(this, "请填写消息标题", Toast.LENGTH_SHORT).show();
                    return;
                }
                if (TextUtils.isEmpty(et_message.getText())) {
                    Toast.makeText(this, "请填写消息内容", Toast.LENGTH_SHORT).show();
                    return;
                }
                String title = et_title.getText().toString();
                String message = et_message.getText().toString();
                sendCounterNotify(title, message); // 发送计时的通知消息
            }
        }
    
        // 发送计时的通知消息
        private void sendCounterNotify(String title, String message) {
            // 发送消息之前要先创建通知渠道,创建代码见MainApplication.java
            // 创建一个跳转到活动页面的意图
            Intent cancelIntent = new Intent(this, MainActivity.class);
            // 创建一个用于页面跳转的延迟意图
            PendingIntent deleteIntent = PendingIntent.getActivity(this,
                    R.string.app_name, cancelIntent, PendingIntent.FLAG_UPDATE_CURRENT);
            // 创建一个通知消息的建造器
            Notification.Builder builder = new Notification.Builder(this);
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
                // Android 8.0开始必须给每个通知分配对应的渠道
                builder = new Notification.Builder(this, getString(R.string.app_name));
            }
            builder.setDeleteIntent(deleteIntent) // 设置内容的清除意图
                    .setSmallIcon(R.mipmap.ic_launcher) // 设置应用名称左边的小图标
                    // 设置通知栏右边的大图标
                    .setLargeIcon(BitmapFactory.decodeResource(getResources(), R.drawable.ic_app))
                    .setProgress(100, 60, false) // 设置进度条及其具体进度
                    .setUsesChronometer(true) // 设置是否显示计时器
                    .setContentTitle(title) // 设置通知栏里面的标题文本
                    .setContentText(message); // 设置通知栏里面的内容文本
            Notification notify = builder.build(); // 根据通知建造器构建一个通知对象
            // 从系统服务中获取通知管理器
            NotificationManager notifyMgr = (NotificationManager)
                    getSystemService(Context.NOTIFICATION_SERVICE);
            // 使用通知管理器推送通知,然后在手机的通知栏就会看到该消息
            notifyMgr.notify(R.string.app_name, notify);
        }
    
    }
    

      

    package com.example.chapter11.util;
    
    import android.annotation.TargetApi;
    import android.app.Notification;
    import android.app.NotificationChannel;
    import android.app.NotificationManager;
    import android.content.Context;
    import android.net.Uri;
    import android.os.Build;
    import android.os.Bundle;
    import android.util.Log;
    
    import com.example.chapter11.BuildConfig;
    
    import java.lang.reflect.Field;
    import java.lang.reflect.Method;
    
    public class NotifyUtil {
        private final static String TAG = "NotifyUtil";
    
        @TargetApi(Build.VERSION_CODES.O)
        // 创建通知渠道。Android 8.0开始必须给每个通知分配对应的渠道
        public static void createNotifyChannel(Context ctx, String channelId, String channelName, int importance) 
    	{
            // 从系统服务中获取通知管理器
            NotificationManager notifyMgr = (NotificationManager)
                    ctx.getSystemService(Context.NOTIFICATION_SERVICE);
            if (notifyMgr.getNotificationChannel(channelId) == null) 
    		{ // 已经存在指定编号的通知渠道
    	
                // 创建指定编号、指定名称、指定级别的通知渠道
                NotificationChannel channel = new NotificationChannel(channelId, channelName, importance);
                channel.setSound(null, null); // 设置推送通知之时的铃声。null表示静音推送
                channel.enableLights(true); // 通知渠道是否让呼吸灯闪烁
                channel.enableVibration(true); // 通知渠道是否让手机震动
                channel.setShowBadge(true); // 通知渠道是否在应用图标的右上角展示小红点
                // VISIBILITY_PUBLIC显示所有通知信息,VISIBILITY_PRIVATE只显示通知标题不显示通知内容,VISIBILITY_SECRET不显示任何通知信息
                channel.setLockscreenVisibility(Notification.VISIBILITY_PRIVATE); // 设置锁屏时候的可见性
                channel.setImportance(importance); // 设置通知渠道的重要性级别
                notifyMgr.createNotificationChannel(channel); // 创建指定的通知渠道
            }
        }
    
        // 在桌面上的应用图标右上角显示数字角标
        public static void showMarkerCount(Context ctx, int count, Notification notify) {
            showBadgeOfEMUI(ctx, count); // 华为手机EMUI系统的消息角标
            // 小米手机还要进入设置里面的应用管理,开启当前App的“显示桌面图标角标”
            showBadgeOfMIUI(count, notify); // 小米手机MIUI系统的消息角标
        }
    
        // 华为的消息角标需要事先声明两个权限:android.permission.INTERNET、com.huawei.android.launcher.permission.CHANGE_BADGE
        private static void showBadgeOfEMUI(Context ctx, int count) {
            try {
                Bundle extra = new Bundle(); // 创建一个包裹对象
                extra.putString("package", BuildConfig.APPLICATION_ID); // 应用的包名
                // 应用的首屏页面路径
                extra.putString("class", BuildConfig.APPLICATION_ID+".MainActivity");
                extra.putInt("badgenumber", count); // 应用的消息数量
                Uri uri = Uri.parse("content://com.huawei.android.launcher.settings/badge/");
                // 通过内容解析器调用华为内核的消息角标服务
                ctx.getContentResolver().call(uri, "change_badge", null, extra);
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    
        // 小米的消息角标需要在发送通知的时候一块调用
        private static void showBadgeOfMIUI(int count, Notification notify) {
            try {
                // 利用反射技术获得额外的新增字段extraNotification
                Field field = notify.getClass().getDeclaredField("extraNotification");
                Log.d(TAG, "field.getName="+field.getName());
                // 该字段为Notification类型,下面获取它的实例对象
                Object extra = field.get(notify);
                Log.d(TAG, "extraNotification.toString="+extra.toString());
                // 利用反射技术获得额外的新增方法setMessageCount
                Method method = extra.getClass().getDeclaredMethod("setMessageCount", int.class);
                Log.d(TAG, "method.getName="+method.getName());
                // 利用反射技术调用实例对象的setMessageCount方法,设置消息角标的数量
                method.invoke(extra, count);
                Log.d(TAG, "invoke count="+count);
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    
    }
    

      

    package com.example.chapter11;
    
    import android.app.Application;
    import android.app.NotificationManager;
    import android.os.Build;
    import android.util.Log;
    
    import com.example.chapter11.util.NotifyUtil;
    
    public class MainApplication extends Application {
        private final static String TAG = "MainApplication";
    
        @Override
        public void onCreate() {
            super.onCreate();
            // 这里不能屏蔽通知渠道代码,因为后面活动会给该渠道发送通知
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
                // Android 8.0开始必须给每个通知分配对应的渠道
                NotifyUtil.createNotifyChannel(this, getString(R.string.app_name), getString(R.string.app_name), NotificationManager.IMPORTANCE_LOW);
            }
            Log.d(TAG, "onCreate");
        }
    
    }
    

      

    package com.example.chapter11;
    
    import android.app.Notification;
    import android.app.NotificationManager;
    import android.app.PendingIntent;
    import android.content.Context;
    import android.content.Intent;
    import android.os.Build;
    import android.os.Bundle;
    import android.text.TextUtils;
    import android.view.View;
    import android.widget.AdapterView;
    import android.widget.ArrayAdapter;
    import android.widget.EditText;
    import android.widget.Spinner;
    import android.widget.Toast;
    
    import androidx.appcompat.app.AppCompatActivity;
    
    import com.example.chapter11.util.NotifyUtil;
    import com.example.chapter11.util.ViewUtil;
    
    public class NotifyChannelActivity extends AppCompatActivity implements View.OnClickListener {
        private EditText et_title;
        private EditText et_message;
        private String mChannelId = "0"; // 通知渠道的编号
        private String mChannelName; // 通知渠道的名称
        private int mImportance; // 通知渠道的级别
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_notify_channel);
            et_title = findViewById(R.id.et_title);
            et_message = findViewById(R.id.et_message);
            findViewById(R.id.btn_send_channel).setOnClickListener(this);
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
                initImportanceSpinner(); // 初始化渠道级别的下拉框
            }
        }
    
        // 初始化渠道级别的下拉框
        private void initImportanceSpinner() {
            findViewById(R.id.ll_channel).setVisibility(View.VISIBLE);
            ArrayAdapter<String> importanceAdapter = new ArrayAdapter<String>(this,
                    R.layout.item_select, importanceDescArray);
            Spinner sp_importance = findViewById(R.id.sp_importance);
            sp_importance.setPrompt("请选择渠道级别");
            sp_importance.setAdapter(importanceAdapter);
            sp_importance.setSelection(3);
            sp_importance.setOnItemSelectedListener(new TypeSelectedListener());
        }
    
        private int[] importanceTypeArray = {NotificationManager.IMPORTANCE_NONE,
                NotificationManager.IMPORTANCE_MIN,
                NotificationManager.IMPORTANCE_LOW,
                NotificationManager.IMPORTANCE_DEFAULT,
                NotificationManager.IMPORTANCE_HIGH,
                NotificationManager.IMPORTANCE_MAX};
    			
        private String[] importanceDescArray = {"不重要", // 无通知
                "最小级别", // 通知栏折叠,无提示声音,无锁屏通知
                "有点重要", // 通知栏展开,无提示声音,有锁屏通知
                "一般重要", // 通知栏展开,有提示声音,有锁屏通知
                "非常重要", // 通知栏展开,有提示声音,有锁屏通知,在屏幕顶部短暂悬浮(有的手机需要在设置页面开启横幅)
                "最高级别" // 通知栏展开,有提示声音,有锁屏通知,在屏幕顶部短暂悬浮(有的手机需要在设置页面开启横幅)
        };
        class TypeSelectedListener implements AdapterView.OnItemSelectedListener {
            public void onItemSelected(AdapterView<?> arg0, View arg1, int arg2, long arg3) {
                mImportance = importanceTypeArray[arg2];
                mChannelId = "" + arg2;
                mChannelName = importanceDescArray[arg2];
            }
    
            public void onNothingSelected(AdapterView<?> arg0) {
            }
        }
    
        // 发送指定渠道的通知消息(包括消息标题和消息内容)
        private void sendChannelNotify(String title, String message) {
            // 创建一个跳转到活动页面的意图
            Intent clickIntent = new Intent(this, MainActivity.class);
            // 创建一个用于页面跳转的延迟意图
            PendingIntent contentIntent = PendingIntent.getActivity(this,
                    R.string.app_name, clickIntent, PendingIntent.FLAG_UPDATE_CURRENT);
            // 创建一个通知消息的建造器
            Notification.Builder builder = new Notification.Builder(this);
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
                // Android 8.0开始必须给每个通知分配对应的渠道
                builder = new Notification.Builder(this, mChannelId);
            }
            builder.setContentIntent(contentIntent) // 设置内容的点击意图
                    .setAutoCancel(true) // 点击通知栏后是否自动清除该通知
                    .setSmallIcon(R.mipmap.ic_launcher) // 设置应用名称左边的小图标
                    .setContentTitle(title) // 设置通知栏里面的标题文本
                    .setContentText(message); // 设置通知栏里面的内容文本
            Notification notify = builder.build(); // 根据通知建造器构建一个通知对象
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
                NotifyUtil.createNotifyChannel(this, mChannelId, mChannelName, mImportance);
            }
            // 从系统服务中获取通知管理器
            NotificationManager notifyMgr = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
            // 使用通知管理器推送通知,然后在手机的通知栏就会看到该消息,多条通知需要指定不同的通知编号
            notifyMgr.notify(Integer.parseInt(mChannelId), notify);
            if (mImportance != NotificationManager.IMPORTANCE_NONE) {
                Toast.makeText(this, "已发送渠道消息", Toast.LENGTH_SHORT).show();
            }
        }
    
        @Override
        public void onClick(View v) {
            if (v.getId() == R.id.btn_send_channel) {
                ViewUtil.hideOneInputMethod(this, et_message); // 隐藏输入法软键盘
                if (TextUtils.isEmpty(et_title.getText())) {
                    Toast.makeText(this, "请填写消息标题", Toast.LENGTH_SHORT).show();
                    return;
                }
                if (TextUtils.isEmpty(et_message.getText())) {
                    Toast.makeText(this, "请填写消息内容", Toast.LENGTH_SHORT).show();
                    return;
                }
                // 发送指定渠道的通知消息(包括消息标题和消息内容)
                sendChannelNotify(et_title.getText().toString(), et_message.getText().toString());
            }
        }
    }
    

      

    package com.example.chapter11;
    
    import android.app.Notification;
    import android.app.NotificationManager;
    import android.app.PendingIntent;
    import android.content.Context;
    import android.content.Intent;
    import android.os.Build;
    import android.os.Bundle;
    import android.text.TextUtils;
    import android.view.View;
    import android.widget.EditText;
    import android.widget.Toast;
    
    import androidx.appcompat.app.AppCompatActivity;
    
    import com.example.chapter11.util.NotifyUtil;
    import com.example.chapter11.util.ViewUtil;
    
    public class NotifyMarkerActivity extends AppCompatActivity implements View.OnClickListener {
        private EditText et_title;
        private EditText et_message;
        private EditText et_count;
        private static String mChannelId = "3"; // 通知渠道的编号
        private static String mChannelName = "一般重要"; // 通知渠道的名称
        private static int mImportance = NotificationManager.IMPORTANCE_DEFAULT; // 通知渠道的级别
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_notify_marker);
            et_title = findViewById(R.id.et_title);
            et_message = findViewById(R.id.et_message);
            et_count = findViewById(R.id.et_count);
            findViewById(R.id.btn_show_marker).setOnClickListener(this);
            findViewById(R.id.btn_clear_marker).setOnClickListener(this);
        }
    
        @Override
        public void onClick(View v) {
            ViewUtil.hideOneInputMethod(this, et_message); // 隐藏输入法软键盘
            if (TextUtils.isEmpty(et_title.getText())) {
                Toast.makeText(this, "请填写消息标题", Toast.LENGTH_SHORT).show();
                return;
            }
            if (TextUtils.isEmpty(et_message.getText())) {
                Toast.makeText(this, "请填写消息内容", Toast.LENGTH_SHORT).show();
                return;
            }
            if (TextUtils.isEmpty(et_count.getText())) {
                Toast.makeText(this, "请填写消息数量", Toast.LENGTH_SHORT).show();
                return;
            }
            String title = et_title.getText().toString();
            String message = et_message.getText().toString();
            int count = Integer.parseInt(et_count.getText().toString());
            if (v.getId() == R.id.btn_show_marker) {
                sendChannelNotify(title, message, count); // 发送指定渠道的通知消息
                Toast.makeText(this, "已显示消息角标,请回到桌面查看", Toast.LENGTH_SHORT).show();
            } else if (v.getId() == R.id.btn_clear_marker) {
                sendChannelNotify(title, message, 0); // 发送指定渠道的通知消息
                Toast.makeText(this, "已清除消息角标,请回到桌面查看", Toast.LENGTH_SHORT).show();
            }
        }
    
        // 发送指定渠道的通知消息(包括消息标题和消息内容)
        private void sendChannelNotify(String title, String message, int count) {
            // 创建一个跳转到活动页面的意图
            Intent clickIntent = new Intent(this, MainActivity.class);
            // 创建一个用于页面跳转的延迟意图
            PendingIntent contentIntent = PendingIntent.getActivity(this,
                    R.string.app_name, clickIntent, PendingIntent.FLAG_UPDATE_CURRENT);
            // 创建一个通知消息的建造器
            Notification.Builder builder = new Notification.Builder(this);
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
                // Android 8.0开始必须给每个通知分配对应的渠道
                builder = new Notification.Builder(this, mChannelId);
            }
            builder.setContentIntent(contentIntent) // 设置内容的点击意图
                    .setAutoCancel(true) // 点击通知栏后是否自动清除该通知
                    .setSmallIcon(R.mipmap.ic_launcher) // 设置应用名称左边的小图标
                    .setContentTitle(title) // 设置通知栏里面的标题文本
                    .setContentText(message); // 设置通知栏里面的内容文本
            Notification notify = builder.build(); // 根据通知建造器构建一个通知对象
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
                NotifyUtil.createNotifyChannel(this, mChannelId, mChannelName, mImportance);
            }
            NotifyUtil.showMarkerCount(this, count, notify); // 在桌面的应用图标右上方显示指定数字的消息角标
            // 从系统服务中获取通知管理器
            NotificationManager notifyMgr = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
            // 使用通知管理器推送通知,然后在手机的通知栏就会看到该消息,多条通知需要指定不同的通知编号
            notifyMgr.notify(Integer.parseInt(mChannelId), notify);
        }
    
    }
    

      

  • 相关阅读:
    go 接收发送文件
    【0031】反转整数/判断回文
    【003】链表或字符串的【反转】【左旋转】
    【002】链表或字符串模拟加法/加一/乘法
    【01】数组中只出现一次的数字
    【面试题050】树中两个结点的最低公共祖先
    【面试题049】把字符串转换成整数
    【面试题048】不能继承的类
    【面试题047】不用加减乘除做加法
    【面试题046】求1+2+...+n
  • 原文地址:https://www.cnblogs.com/xiaobaibailongma/p/15733269.html
Copyright © 2011-2022 走看看