1.AndroidStudio离线打包MUI
如何离线打包请参看上篇随笔《AndroidStudio离线打包MUI》
2.集成极光推送
官方文档:https://docs.jiguang.cn/jpush/client/Android/android_guide/
建议采用 jcenter 自动集成 的方式,手动集成对新手来说容易出错
使用jcenter自动集成的开发者,不需要在项目中添加jar和so,jcenter会自动完成依赖;在AndroidManifest.xml中不需要添加任何JPush SDK 相关的配置,jcenter会自动导入。
-
如果开发者需要修改组件属性,可以在本地的 AndroidManifest 中定义同名的组件并配置想要的属性,然后用 xmlns:tools 来控制本地组件覆盖 jcenter 上的组件。示例:
<manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.android.tests.flavorlib.app" xmlns:tools="http://schemas.android.com/tools"> <application android:icon="@drawable/icon" android:name="com.example.jpushdemo.ExampleApplication" android:label="@string/app_name" > <service android:name="cn.jpush.android.service.PushService" android:process=":multiprocess" tools:node="replace" > …… </service> …… </application> …… </manifest>
-
确认android studio的 Project 根目录的主 gradle 中配置了jcenter支持。(新建project默认配置就支持)
buildscript { repositories { jcenter() } ...... } allprojets { repositories { jcenter() } }
-
在 module 的 gradle 中添加依赖和AndroidManifest的替换变量。
android { ...... defaultConfig { applicationId "com.xxx.xxx" //JPush上注册的包名. ...... ndk { //选择要添加的对应cpu类型的.so库。 abiFilters 'armeabi', 'armeabi-v7a', 'arm64-v8a' // 还可以添加 'x86', 'x86_64', 'mips', 'mips64' } manifestPlaceholders = [ JPUSH_PKGNAME : applicationId, JPUSH_APPKEY : "你的appkey", //JPush上注册的包名对应的appkey. JPUSH_CHANNEL : "developer-default", //暂时填写默认值即可. ] ...... } ...... } dependencies { ...... compile 'cn.jiguang.sdk:jpush:3.1.1' // 此处以JPush 3.1.1 版本为例。 compile 'cn.jiguang.sdk:jcore:1.1.9' // 此处以JCore 1.1.9 版本为例。 ...... }
3.Android项目中增加注册、获取极光推送相关信息的代码
3.1.在AS中增加JPushUtil工具类
import android.content.Context;
import android.content.pm.ApplicationInfo;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager;
import android.content.pm.PackageManager.NameNotFoundException;
import android.net.ConnectivityManager;
import android.net.NetworkInfo;
import android.os.Bundle;
import android.os.Looper;
import android.telephony.TelephonyManager;
import android.text.TextUtils;
import android.widget.Toast;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import cn.jpush.android.api.JPushInterface;
import io.dcloud.common.adapter.util.Logger;
public class JpushUtil {
public static final String PREFS_NAME = "JPUSH_EXAMPLE";
public static final String PREFS_DAYS = "JPUSH_EXAMPLE_DAYS";
public static final String PREFS_START_TIME = "PREFS_START_TIME";
public static final String PREFS_END_TIME = "PREFS_END_TIME";
public static final String KEY_APP_KEY = "JPUSH_APPKEY";
public static boolean isEmpty(String s) {
if (null == s)
return true;
if (s.length() == 0)
return true;
if (s.trim().length() == 0)
return true;
return false;
}
/**
* 只能以 “+” 或者 数字开头;后面的内容只能包含 “-” 和 数字。
* */
private final static String MOBILE_NUMBER_CHARS = "^[+0-9][-0-9]{1,}$";
public static boolean isValidMobileNumber(String s) {
if(TextUtils.isEmpty(s)) return true;
Pattern p = Pattern.compile(MOBILE_NUMBER_CHARS);
Matcher m = p.matcher(s);
return m.matches();
}
// 校验Tag Alias 只能是数字,英文字母和中文
public static boolean isValidTagAndAlias(String s) {
Pattern p = Pattern.compile("^[\u4E00-\u9FA50-9a-zA-Z_!@#$&*+=.|]+$");
Matcher m = p.matcher(s);
return m.matches();
}
// 取得AppKey
public static String getAppKey(Context context) {
Bundle metaData = null;
String appKey = null;
try {
ApplicationInfo ai = context.getPackageManager().getApplicationInfo(
context.getPackageName(), PackageManager.GET_META_DATA);
if (null != ai)
metaData = ai.metaData;
if (null != metaData) {
appKey = metaData.getString(KEY_APP_KEY);
if ((null == appKey) || appKey.length() != 24) {
appKey = null;
}
}
} catch (NameNotFoundException e) {
}
return appKey;
}
// 取得版本号
public static String GetVersion(Context context) {
try {
PackageInfo manager = context.getPackageManager().getPackageInfo(
context.getPackageName(), 0);
return manager.versionName;
} catch (NameNotFoundException e) {
return "Unknown";
}
}
public static void showToast(final String toast, final Context context)
{
new Thread(new Runnable() {
@Override
public void run() {
Looper.prepare();
Toast.makeText(context, toast, Toast.LENGTH_SHORT).show();
Looper.loop();
}
}).start();
}
public static boolean isConnected(Context context) {
ConnectivityManager conn = (ConnectivityManager) context.getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo info = conn.getActiveNetworkInfo();
return (info != null && info.isConnected());
}
public static String getImei(Context context, String imei) {
String ret = null;
try {
TelephonyManager telephonyManager = (TelephonyManager) context.getSystemService(Context.TELEPHONY_SERVICE);
ret = telephonyManager.getDeviceId();
} catch (Exception e) {
Logger.e(JpushUtil.class.getSimpleName(), e.getMessage());
}
if (isReadableASCII(ret)){
return ret;
} else {
return imei;
}
}
private static boolean isReadableASCII(CharSequence string){
if (TextUtils.isEmpty(string)) return false;
try {
Pattern p = Pattern.compile("[\\x20-\\x7E]+");
return p.matcher(string).matches();
} catch (Throwable e){
return true;
}
}
public static String getDeviceId(Context context) {
return JPushInterface.getUdid(context);
}
}
3.2.增加JPushInitActivity类
import android.os.Bundle;
import cn.jpush.android.api.InstrumentedActivity;
import cn.jpush.android.api.JPushInterface;
/**
* 初始化极光推送的相关信息
*/
public class JPushInitActivity extends InstrumentedActivity {
public static String APP_KEY = "";//在极光推送中注册的应用ID
public static String MASTER_SECRET = "08123213666d973dkkik3bbe7fd6";//在极光推送官网注册后获得的密码(请改为你自己注册后的值)
public static String REGISTRATION_ID = "";//安装APP的用户在极光推送中注册的ID
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
init();
}
public void init(){
JPushInterface.init(getApplicationContext());
//获取初始化之后的环境
APP_KEY = JpushUtil.getAppKey(getApplicationContext());
REGISTRATION_ID = JPushInterface.getRegistrationID(getApplicationContext());
}
public static String getJPushData(){
return "appKey:"+APP_KEY+";masterSecret:"+MASTER_SECRET+";registrationId:"+REGISTRATION_ID;
}
}
3.3.在AndroidManifest.xml中注册JPushInitActivity
<!-- 用于初始化极光推送的注册数据,不显示给用户 -->
<activity
android:name=".jpush.JPushInitActivity"
android:theme="@android:style/Theme.NoDisplay">
</activity>
3.4.在mui.js中创建Activity并完成注册
mui.plusReady(function() {
//调用原生Activity
var Intent = plus.android.importClass("android.content.Intent");
// 获取主Activity对象的实例
var main = plus.android.runtimeMainActivity();
// 创建Intent
var naviIntent = new Intent();
var ComponentName = plus.android.importClass("android.content.ComponentName");
//创建极光推送注册Activity,包名请换成你自己的包名
naviIntent.setComponent(new ComponentName(main, "com.xxx.xxx.xxx.JPushInitActivity"));
main.startActivity(naviIntent);
//调用java方法
$(".login_btn").click(function(){
//引入java类文件,包名请换成你自己的包名
var jPush = plus.android.importClass("com.xxx.xxx.xxx.JPushInitActivity");
//调用静态方法
var jPushData = jPush.getJPushData();
//输出返回值
alert(jPushData);
//TODO 在登录时将jPushData及用户名、密码一并传给java后端
//......
})
})
4.java端代码
下载极光推送的jar包并引入到项目: https://github.com/jpush/jpush-api-java-client/releases
4.1 接收登录信息,将registrationId与用户绑定,请根据自己的业务去进行关联
//验证成功,将jPushId与用户关联起来
try {
String[] split = jPushData.split(";");
for (String str : split) {
if ("registrationId".equals(str.split(":")[0])) {
String jPushId = str.split(":")[1];
String sql = "UPDATE BO_PBS_ORG_EXT_USER SET J_PUSH_ID = ? WHERE YHZH = ?";
Object[] params = { jPushId, loginUserModel.getUSER_ACCOUNT() };
db.update(sql, params);
break;
}
}
} catch (Exception e) {
e.printStackTrace();
}
4.2 新增service接口JpushMessageService
public interface JpushMessageService {
/**
* 发送给所有用户
* @param content
* @return
*/
public String sendPushAll(String content);
/**
* 发送给userId对应的用户
* @param userId
* @param content
* @return
*/
public String senPushByUserId(String userId, String content);
}
4. 3 新增service接口实现类JpushMessageServiceImpl
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import javax.annotation.Resource;
import org.apache.commons.lang.StringUtils;
import org.apache.log4j.Logger;
import org.springframework.stereotype.Service;
import cn.jiguang.common.ClientConfig;
import cn.jiguang.common.resp.APIConnectionException;
import cn.jiguang.common.resp.APIRequestException;
import cn.jpush.api.JPushClient;
import cn.jpush.api.push.PushResult;
import cn.jpush.api.push.model.PushPayload;
@Service
public class JpushMessageServiceImpl implements JpushMessageService {
@Resource
UserService userService;
@Resource
DataBaseService db;
private final static String appKey = "weqwewe1123123123123";//这里请填写自己注册获得的值,或者通过mui向后台传值
private final static String masterSecret = "213123123asdjfoi1293";//这里请填写自己注册获得的值,或者通过mui向后台传值
/**
* 保存离线的时长。秒为单位。最多支持10天(864000秒)。 0 表示该消息不保存离线。即:用户在线马上发出,当前不在线用户将不会收到此消息。
* 此参数不设置则表示默认,默认为保存1天的离线消息(86400秒)。
*/
private static long timeToLive = 60 * 60 * 24;
private static JPushClient jPushClient = null;
private static final Logger logger = Logger.getLogger(JpushMessageServiceImpl.class);
@Override
public String sendPushAll(String content) {
ClientConfig config = ClientConfig.getInstance();
config.setMaxRetryTimes(5);
config.setConnectionTimeout(10 * 1000);
config.setGlobalPushSetting(false, timeToLive);
jPushClient = new JPushClient(masterSecret, appKey, null, config);
boolean flag = false;
try {
PushPayload payload = JPushUtil.buildPushObject_all_all_alert(content);
PushResult result = jPushClient.sendPush(payload);
if (null != result) {
logger.info("Get resul ---" + result);
flag = true;
}
} catch (APIConnectionException e) {
logger.error("Connection error. Should retry later. ", e);
} catch (APIRequestException e) {
logger.error("Error response from JPush server. Should review and fix it. ", e);
logger.info("HTTP Status: " + e.getStatus());
logger.info("Error Code: " + e.getErrorCode());
logger.info("Error Message: " + e.getErrorMessage());
logger.info("Msg ID: " + e.getMsgId());
}
Map<String, Object> result = new HashMap<String, Object>();
if (flag) {
result.put("status", "ok");
result.put("code", "0");
result.put("msg", "发送成功");
} else {
result.put("status", "fail");
result.put("code", "-1");
result.put("msg", "发送失败");
}
return ReturnUtil.getJsonStr(result);
}
@Override
public String senPushByUserId(String userId, String content) {
boolean flag = false;
try {
//在数据库中查询极光推送注册信息以及是否接受推送(可以根据自己业务省略是否接收推送的判断)
//eu.J_PUSH_ID就是我们在登录之后跟用户管理起来的registrationId
String sql = "SELECT eu.J_PUSH_ID,eu.ACCEPT_PUSH FROM PBS_ORG_USER u JOIN BO_PBS_ORG_EXT_USER eu ON u.USER_ACCOUNT = eu.YHZH AND u.ID = ?";
Object[] params = { userId };
List<Map<String, Object>> records = db.queryList(sql, params);
if (records != null && records.size() > 0) {
Boolean acceptPush = records.get(0).get("ACCEPT_PUSH") == null ? true : (boolean) records.get(0).get("ACCEPT_PUSH");
String jPushId = records.get(0).get("J_PUSH_ID") == null ? "" : (String) records.get(0).get("J_PUSH_ID");
if (acceptPush && StringUtils.isNotEmpty(jPushId)) {
JPushClient jPushClient = new JPushClient(masterSecret, appKey);
List<String> regeSterIds = new ArrayList<>();
regeSterIds.add(jPushId);
try {
PushPayload payload = JPushUtil.buildPushObject_all_all_regesterIds(regeSterIds, content);
PushResult pushResult = jPushClient.sendPush(payload);
if (null != pushResult) {
logger.info("Get result ----" + pushResult);
flag = true;
}
} catch (APIConnectionException e) {
logger.error("Connection error. Should retry later. ", e);
} catch (APIRequestException e) {
logger.error("Error response from JPush server. Should review and fix it. ", e);
logger.info("HTTP Status: " + e.getStatus());
logger.info("Error Code: " + e.getErrorCode());
logger.info("Error Message: " + e.getErrorMessage());
logger.info("Msg ID: " + e.getMsgId());
}
}
}
} catch (Exception e) {
e.printStackTrace();
}
Map<String, Object> result = new HashMap<String, Object>();
if (flag) {
result.put("status", "ok");
result.put("code", "0");
result.put("msg", "发送成功");
} else {
result.put("status", "fail");
result.put("code", "-1");
result.put("msg", "发送失败");
}
return ReturnUtil.getJsonStr(result);
}
}