gpt4 book ai didi

android - 在 Android KitKat 中接收彩信

转载 作者:可可西里 更新时间:2023-11-01 18:45:15 29 4
gpt4 key购买 nike

所以这个视频Android 4.4 SMS APIs来自#DevBytes 的文章解释了 KitKat 中 SMS API 的最新变化。他们还提供了示例项目的链接。 http://goo.gl/uQ3Nih

他们建议您在服务中处理 MMS 的接收。一切看起来都很好,除了他们忽略了最没有记录的部分。如何实际处理传入的彩信。

这是项目的示例 https://gist.github.com/lawloretienne/8970938

我试过“处理彩信”

https://gist.github.com/lawloretienne/8971050

我可以从 Intent 中获取额外信息,但我可以提取的唯一有意义的东西是发送彩信的号码。

谁能为我指出正确的方向,告诉我如何去做?

我注意到 WAP_PUSH_MESSAGE 包含一些内容,FROM、SUBJECT 和 CONTENT_LOCATION。

内容位置似乎是包含 MMS 内容的 url。我怎样才能访问它?

这是该 URL 的示例

https://atl1mmsget.msg.eng.t-mobile.com/mms/wapenc?location=XXXXXXXXXXX_14zbwk&rid=027

其中 X 是我正在测试的设备的电话号码中的一个数字。

看起来美国 T-Mobile 的 MMSC(多媒体消息服务中心)是 http://mms.msg.eng.t-mobile.com/mms/wapenc

根据此列表:http://www.activexperts.com/xmstoolkit/mmsclist/

最佳答案

文档为零,因此这里有一些信息可以提供帮助。

1) 来自源代码的 com.google.android.mms.pdu。您需要 Pdu 实用程序。

2) 您从传入的 mms 广播的额外字节数组中获得通知推送 (intent.getByteArrayExtra("data"))。

3) 将通知推送解析为 GenericPdu (new PduParser(rawPdu).parse())。

4) 您需要 TransactionSettings 才能与运营商的 wap 服务器进行通信。我在下面#5 之后获得交易设置。我使用:

TransactionSettings transactionSettings = new TransactionSettings(mContext, mConnMgr.getNetworkInfo(ConnectivityManager.TYPE_MOBILE_MMS).getExtraInfo());

5) 强制通过 wifi 进行网络通信。我使用以下内容。

private boolean beginMmsConnectivity() {
try {
int result = mConnMgr.startUsingNetworkFeature(ConnectivityManager.TYPE_MOBILE, Phone.FEATURE_ENABLE_MMS);
NetworkInfo info = mConnMgr.getNetworkInfo(ConnectivityManager.TYPE_MOBILE_MMS);
boolean isAvailable = info != null && info.isConnected() && result == Phone.APN_ALREADY_ACTIVE && !Phone.REASON_VOICE_CALL_ENDED.equals(info.getReason());
return isAvailable;
} catch(Exception e) {
return false;
}
}

6) 然后您需要确保到主机的路由。

private static void ensureRouteToHost(ConnectivityManager cm, String url, TransactionSettings settings) throws IOException {
int inetAddr;
if (settings.isProxySet()) {
String proxyAddr = settings.getProxyAddress();
inetAddr = lookupHost(proxyAddr);
if (inetAddr == -1) {
throw new IOException("Cannot establish route for " + url + ": Unknown host");
} else {
if (!cm.requestRouteToHost(ConnectivityManager.TYPE_MOBILE_MMS, inetAddr))
throw new IOException("Cannot establish route to proxy " + inetAddr);
}
} else {
Uri uri = Uri.parse(url);
inetAddr = lookupHost(uri.getHost());
if (inetAddr == -1) {
throw new IOException("Cannot establish route for " + url + ": Unknown host");
} else {
if (!cm.requestRouteToHost(ConnectivityManager.TYPE_MOBILE_MMS, inetAddr))
throw new IOException("Cannot establish route to " + inetAddr + " for " + url);
}
}
}

这是 lookupHost 方法:

private static int lookupHost(String hostname) {
InetAddress inetAddress;
try {
inetAddress = InetAddress.getByName(hostname);
} catch (UnknownHostException e) {
return -1;
}
byte[] addrBytes;
int addr;
addrBytes = inetAddress.getAddress();
addr = ((addrBytes[3] & 0xff) << 24) | ((addrBytes[2] & 0xff) << 16) | ((addrBytes[1] & 0xff) << 8) | (addrBytes[0] & 0xff);
return addr;
}

我还喜欢使用基于反射的方法来改进 ensureRouteToHost 功能:

private static void ensureRouteToHostFancy(ConnectivityManager cm, String url, TransactionSettings settings) throws IOException, NoSuchMethodException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {
Method m = cm.getClass().getMethod("requestRouteToHostAddress", new Class[] { int.class, InetAddress.class });
InetAddress inetAddr;
if (settings.isProxySet()) {
String proxyAddr = settings.getProxyAddress();
try {
inetAddr = InetAddress.getByName(proxyAddr);
} catch (UnknownHostException e) {
throw new IOException("Cannot establish route for " + url + ": Unknown proxy " + proxyAddr);
}
if (!(Boolean) m.invoke(cm, new Object[] { ConnectivityManager.TYPE_MOBILE_MMS, inetAddr }))
throw new IOException("Cannot establish route to proxy " + inetAddr);
} else {
Uri uri = Uri.parse(url);
try {
inetAddr = InetAddress.getByName(uri.getHost());
} catch (UnknownHostException e) {
throw new IOException("Cannot establish route for " + url + ": Unknown host");
}
if (!(Boolean) m.invoke(cm, new Object[] { ConnectivityManager.TYPE_MOBILE_MMS, inetAddr }))
throw new IOException("Cannot establish route to " + inetAddr + " for " + url);
}
}

7) 在确保到主机的路由后,您可以从源中获取 HttpUtls。我使用 OkHttp 对我的实现进行了大量修改以改进通信。

byte[] rawPdu = HttpUtils.httpConnection(mContext, mContentLocation, null, HttpUtils.HTTP_GET_METHOD, mTransactionSettings.isProxySet(), mTransactionSettings.getProxyAddress(), mTransactionSettings.getProxyPort());

8) 从生成的字节数组中使用 PduParser 来解析 GenericPdu。然后您可以提取正文并转换为 MultimediaMessagePdu。

9) 然后你可以迭代PDU的部分。

使用 MMS 需要考虑的事情数不胜数。我想到的一件事是幻灯片有多烦人,所以我要做的是检测 PDU 中是否有超过 1 个部分,然后我复制标题并创建单独的 MultimediaMessagePdu,我将它们分别保存到手机的彩信内容提供程序.不要忘记复制标题,特别是如果您支持群发消息。群组消息传递是另一回事,因为 PDU 中的传入电话号码并不能说明全部情况 (MultitimediaMessagePdu.mmpdu())。您使用以下代码提取的 header 中有更多联系人。

private HashSet<String> getRecipients(GenericPdu pdu) {
PduHeaders header = pdu.getPduHeaders();
HashMap<Integer, EncodedStringValue[]> addressMap = new HashMap<Integer, EncodedStringValue[]>(ADDRESS_FIELDS.length);
for (int addrType : ADDRESS_FIELDS) {
EncodedStringValue[] array = null;
if (addrType == PduHeaders.FROM) {
EncodedStringValue v = header.getEncodedStringValue(addrType);
if (v != null) {
array = new EncodedStringValue[1];
array[0] = v;
}
} else {
array = header.getEncodedStringValues(addrType);
}
addressMap.put(addrType, array);
}
HashSet<String> recipients = new HashSet<String>();
loadRecipients(PduHeaders.FROM, recipients, addressMap, false);
loadRecipients(PduHeaders.TO, recipients, addressMap, true);
return recipients;
}

这是加载收件人的方法:

private void loadRecipients(int addressType, HashSet<String> recipients, HashMap<Integer, EncodedStringValue[]> addressMap, boolean excludeMyNumber) {
EncodedStringValue[] array = addressMap.get(addressType);
if (array == null) {
return;
}
// If the TO recipients is only a single address, then we can skip loadRecipients when
// we're excluding our own number because we know that address is our own.
if (excludeMyNumber && array.length == 1) {
return;
}
String myNumber = excludeMyNumber ? mTelephonyManager.getLine1Number() : null;
for (EncodedStringValue v : array) {
if (v != null) {
String number = v.getString();
if ((myNumber == null || !PhoneNumberUtils.compare(number, myNumber)) && !recipients.contains(number)) {
// Only add numbers which aren't my own number.
recipients.add(number);
}
}
}
}

下面是如何迭代 MultimediaMessagePdu 部分。

private void processPduAttachments() throws Exception {
if (mGenericPdu instanceof MultimediaMessagePdu) {
PduBody body = ((MultimediaMessagePdu) mGenericPdu).getBody();
if (body != null) {
int partsNum = body.getPartsNum();
for (int i = 0; i < partsNum; i++) {
try {
PduPart part = body.getPart(i);
if (part == null || part.getData() == null || part.getContentType() == null || part.getName() == null)
continue;
String partType = new String(part.getContentType());
String partName = new String(part.getName());
Log.d("Part Name: " + partName);
Log.d("Part Type: " + partType);
if (ContentType.isTextType(partType)) {
} else if (ContentType.isImageType(partType)) {
} else if (ContentType.isVideoType(partType)) {
} else if (ContentType.isAudioType(partType)) {
}
} catch (Exception e) {
e.printStackTrace();
// Bad part shouldn't ruin the party for the other parts
}
}
}
} else {
Log.d("Not a MultimediaMessagePdu PDU");
}
}

还有更多考虑因素,例如动画 GIF 支持,这是完全可能的:)一些运营商也支持确认报告和交付报告,您很可能会忽略这些 wap 通信,除非用户真的非常想要 mms 交付报告。

关于android - 在 Android KitKat 中接收彩信,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21748209/

29 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com