Android实现彩信附件的添加与删除功能

本文实例讲述了Android实现彩信附件的添加与删除功能。分享给大家供大家参考,具体如下:

添加附件

在ComposeMessageActivity里

addAttachment(int type) 函数

根据type的不同,分成6个case

case A:

MediaSelectListActivity.ADD_IMAGE 用gallery选图片:

MessageUtils.selectImage(this, REQUEST_CODE_ATTACH_IMAGE);

起一个intent:

Intent innerIntent = new Intent(Intent.ACTION_GET_CONTENT);
innerIntent.setType(contentType); //image type
Intent wrapperIntent = Intent.createChooser(innerIntent, null);
startActivityForResult(wrapperIntent,requestCode);

createChooser 函数new 一个 Intent intent = new Intent(ACTION_CHOOSER);

也就是起来一个ACTION_CHOOSER的 activity

case B:

MediaSelectListActivity.TAKE_PICTURE

Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
intent.putExtra(MediaStore.EXTRA_OUTPUT, Mms.ScrapSpace.CONTENT_URI);
startActivityForResult(intent, REQUEST_CODE_TAKE_PICTURE);

起一个照相机来拍照了,

case C:

MediaSelectListActivity.ADD_VIDEO

很像case A

只有在type那里,从image换成video

case D:

MediaSelectListActivity.RECORD_VIDEO

和case B一样起一个录像机,不过这次有空间大小计算:给文本留1024Byte。

按一条彩信300k算,录像最多使用299k。也即299*1024byte

Intent intent = new Intent(MediaStore.ACTION_VIDEO_CAPTURE);
startActivityForResult(intent, REQUEST_CODE_TAKE_VIDEO);

case E:

MediaSelectListActivity.ADD_SOUND

MessageUtils.selectAudio(this, REQUEST_CODE_ATTACH_SOUND);
Intent intent = new Intent(RingtoneManager.ACTION_RINGTONE_PICKER);

case F:

MediaSelectListActivity.RECORD_SOUND

像B那样,七个录音机。 这次type是aution

Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
intent.setType(ContentType.AUDIO_AMR);
intent.setClassName("com.android.soundrecorder", "com.android.soundrecorder.SoundRecorder");

case G:

MediaSelectListActivity.ADD_SLIDESHOW

幻灯片slideshow比较尴尬。因为整个彩信附件也叫slideshow,而这里的slideshow是多张图片拼凑在一起的意思。彩信的附件也经常是若干张图片。。。 这里从起名开始就绕的要死了。

Uri dataUri = mWorkingMessage.saveAsMms(false);
Intent intent = new Intent(this, SlideshowEditActivity.class);
intent.setData(dataUri);
startActivityForResult(intent, REQUEST_CODE_CREATE_SLIDESHOW);

saveAsMms函数里面有几个事情:首先把短信强制变成彩信,把短信的内容封装到PduPersister中(可以理解为彩信body)去,new一个SendReq也就是彩信头。生成包含彩信头和body的uri。
最后起一个尴尬的SlideshowEditActivity。 这个东西太麻烦。不看。。

然后是到了onActivityResult函数。 那些从外面的activity读取媒体数据的应用返回的时候都回到这里。

A。图片

图片在onActivityReuslt会得到uri,调用:

addImage(uri, false);

这个函数会调用:

代码如下:

mWorkingMessage.setAttachment(WorkingMessage.IMAGE, uri, false);

也就是非append模式添加图片。

如果图片太大,会启用异步的图片压缩函数。

B 拍照片

也是返回一个uri到onActivityReuslt函数。也是调用 addImage(uri, false),同上。

C 选录像文件 D 拍录像

都是调用:

addVideo(data.getData(), false);

data.getData()也是得到uri,addVideo调用:

代码如下:

mWorkingMessage.setAttachment(WorkingMessage.VIDEO, uri, append);

跟图片处理一样的。

E 选录音文件 F 录音

都是addAudio -->
mWorkingMessage.setAttachment(WorkingMessage.AUDIO, uri, false);
不多说

删除附件

AttachmentEditor里面有个handler,用来给composeMessageActivity传消息。

所有删除附件操作的按钮都在AttachmentEditor上。对不同的媒体类型有不同的按钮,但是按下之后出口是一样的:

Message msg = Message.obtain(mHandler, MSG_REMOVE_ATTACHMENT);
msg.sendToTarget();

就是这么个操作。

之所以一样是因为所有的附近都存在SlideshowModel里,而这个SlideshowModel是:

ArrayList<SlideModel> mSlides; 一列儿slide组成的。

每个slide可以包含video,image,audio,text, 其中前三者一般不能两两同时存在,唯一的例外是image和audio。

(其实我觉得如果,如果每个slide只能包含三者中的一种,即不处理上面那个例外,逻辑可能更清晰)
说回到那个remove操作。。

composeMessageActivity的Handler里的handleMessage函数,接到这个删除msg之后的操作是:

mWorkingMessage.setAttachment(WorkingMessage.TEXT, null, false);

最后一位false表示非append模式,即从新修改附件。

mWorkingMessage是什么呢?

是短信(包括彩信)的所有的状态所有操作所有数据的现状,主要有几个成员:

mMmsState 彩信状态,是不是彩信,为啥是彩信,是有附件有标题,或者强制彩信等

mAttachmentType 附件类型。如果mSlideshow是多页:slide类型。 单页:图片|声音|录像|文本。   如果mSlideshow是空,就是文本类型。

mSlideshow 附件数据数组。就是那个ArrayList<SlideModel> mSlides。

现在回头看那个删除操作。

setAttachment里面最主要的函数是changeMedia(type, dataUri),这里传入的参数type是TEXT,dataUri是null。
这个函数进去:

SlideModel slide = mSlideshow.get(0);
MediaModel media;
// Remove any previous attachments.
slide.removeImage();
slide.removeVideo();
slide.removeAudio();
// If we're changing to text, just bail out.
if (type == TEXT) {
  return;
}
// Make a correct MediaModel for the type of attachment.
if (type == IMAGE) {
  media = new ImageModel(mContext, uri, mSlideshow.getLayout()
      .getImageRegion());
} else if (type == VIDEO) {
  media = new VideoModel(mContext, uri, mSlideshow.getLayout()
      .getImageRegion());
} else if (type == AUDIO) {
  media = new AudioModel(mContext, uri);
} else {
  throw new IllegalArgumentException("changeMedia type=" + type
      + ", uri=" + uri);
}
// Add it to the slide.
slide.add(media);
// For video and audio, set the duration of the slide to
// that of the attachment.
if (type == VIDEO || type == AUDIO) {
  slide.updateDuration(media.getDuration());
}

看到第一个return我们就可以return了。。

多干净利落!直接把彩信原来的附件看都不堪直接一刀删完,类型回归到WorkingMessage.TEXT,把uri置空。

另外,说一些题外的。

这个changeMedia函数,来来回回,都是改的mSlideshow.get(0)那个

在setAttachment的时候,如果是用的append模式,到时候就会用appendMedia而不是changeMedia函数。

对于append模式,

如果最后一页包含了图片image或者录像vedio, 那么append的时候必须加到下一张。

感觉源码里这个判断有点写复杂了。。你看我一句话就能归纳出来的,他代码写半天~

不过我写不出更好的~~

还有SlideModel的add函数。 很多情况叠在一起了,所以有点复杂。

添加的关键函数是下面这个,第一个参数是对应格式的原来媒体(比如你想添加录像,那这个就是原来的录像,可以是null),第二个是添加的新媒体

private void internalAddOrReplace(MediaModel old, MediaModel media) {
  int addSize = media.getMediaSize();
  int removeSize;
  if (old == null) {
    if (null != mParent) {
      mParent.checkMessageSize(addSize);
    }
    mMedia.add(media);
    increaseSlideSize(addSize);
    increaseMessageSize(addSize);
  } else {
    removeSize = old.getMediaSize();
    if (addSize > removeSize) {
      if (null != mParent) {
        mParent.checkMessageSize(addSize - removeSize);
      }
      increaseSlideSize(addSize - removeSize);
      increaseMessageSize(addSize - removeSize);
    } else {
      decreaseSlideSize(removeSize - addSize);
      decreaseMessageSize(removeSize - addSize);
    }
    mMedia.set(mMedia.indexOf(old), media);
    old.unregisterAllModelChangedObservers();
  }
  for (IModelChangedObserver observer : mModelChangedObservers) {
    media.registerModelChangedObserver(observer);
  }
}

还有个附件太大时的异步缩小功能,是下面这个函数

public static void resizeImageAsync(final Context context,
     final Uri imageUri, final Handler handler,
     final ResizeImageResultCallback cb, final boolean append) {
   // Show a progress toast if the resize hasn't finished
   // within one second.
   // Stash the runnable for showing it away so we can cancel
   // it later if the resize completes ahead of the deadline.
   final Runnable showProgress = new Runnable() {
     public void run() {
       Toast.makeText(context, R.string.compressing,
           Toast.LENGTH_SHORT).show();
     }
   };
   // Schedule it for one second from now.
   handler.postDelayed(showProgress, 1000);
   new Thread(new Runnable() {
     public void run() {
       final PduPart part;
       try {
         UriImage image = new UriImage(context, imageUri);
         part = image.getResizedImageAsPart(MmsConfig
             .getMaxImageWidth(), MmsConfig.getMaxImageHeight(),
             MmsConfig.getMaxMessageSize() - MESSAGE_OVERHEAD);
       } finally {
         // Cancel pending show of the progress toast if necessary.
         handler.removeCallbacks(showProgress);
       }
       handler.post(new Runnable() {
         public void run() {
           cb.onResizeResult(part, append);
         }
       });
     }
   }).start();
}

图片被缩放到最大640*480,如果还是大于300*1024-5000字节(差不多295k),那么缩放到295k。

这个大小是源码编写程序员凭感觉写死的。

这里的cb.onResizeResult是调用的ComposeMessageActivity的ResizeImageResultCallback里的函数。

处理大小结束后,会拿新的图片去再次setAttachment,也就更新了附件。

更多关于Android相关内容感兴趣的读者可查看本站专题:《Android数据库操作技巧总结》、《Android编程之activity操作技巧总结》、《Android文件操作技巧汇总》、《Android编程开发之SD卡操作方法汇总》、《Android开发入门与进阶教程》、《Android资源操作技巧汇总》、《Android视图View技巧总结》及《Android控件用法总结》

希望本文所述对大家Android程序设计有所帮助。

(0)

相关推荐

  • Android下通过httpClient发送GET和POST请求的实例代码

    复制代码 代码如下: public class HttpUtil { public static String sendDataByHttpClientGet(String path,String name,String pass){        String result = "";        //1.获取到一个浏览器        HttpClient client = new DefaultHttpClient();        //2.准备请求的地址        tr

  • Android实现将已发送的短信写入短信数据库的方法

    短信是手机常见的功能,本文就以实例形式讲述了Android实现将已发送的短信写入短信数据库的方法.分享给大家供大家参考之用.具体如下: 一般来说,把短信发送出去以后,需要把已发送的短信写入短信数据库.短信数据库有多个Uri,其中已发送的Uri是content://sms/sent. 具体功能代码如下: // 把短信写入数据库 public void writeMsg(){ try{ ContentValues values = new ContentValues(); // 发送时间 value

  • android中可以通过两种方式调用接口发送短信

    第一:调用系统短信接口直接发送短信:主要代码如下: 复制代码 代码如下: //直接调用短信接口发短信 SmsManager smsManager = SmsManager.getDefault(); List<String> divideContents = smsManager.divideMessage(content); for (String text : divideContents) { smsManager.sendTextMessage("150xxxxxxxx&qu

  • Android编程之非调用系统界面实现发送彩信的方法(MMS)

    本文实例讲述了Android非调用系统界面实现发送彩信的方法.分享给大家供大家参考,具体如下: 一.问题: 最近有个需求,不去调用系统界面发送彩信功能.做过发送短信功能的同学可能第一反应是这样: 不使用 StartActivity,像发短信那样,调用一个类似于发短信的方法 SmsManager smsManager = SmsManager.getDefault(); smsManager.sendTextMessage(phoneCode, null, text, null, null); 二

  • 简单掌握Android开发中彩信的发送接收及其附件的处理

    一. 彩信发送:   彩信比短信麻烦很多.从sendMmsWorker函数的参数就可以看出来:(conv, mmsUri, persister, slideshow, sendReq) 上下文,uri,PduPersister(彩信是用pdu的),slideshow 包含了所有的彩信信息,sendreq包含了mime封装mms时的headers,也包括了from,to等信息 . (1)函数 sendMmsWorker: private void sendMmsWorker(Conversatio

  • Android中发送Http请求(包括文件上传、servlet接收)的实例代码

    复制代码 代码如下: /*** 通过http协议提交数据到服务端,实现表单提交功能,包括上传文件* @param actionUrl 上传路径 * @param params 请求参数 key为参数名,value为参数值 * @param file 上传文件 */public static void postMultiParams(String actionUrl, Map<String, String> params, FormBean[] files) {try {PostMethod p

  • Android实现带附件的邮件发送功能

    本文实例讲解了基于基于JMail实现Android邮件发送功能,分享给大家供大家参考,具体内容如下 在android上发送邮件方式: 第一种:借助GMail APP客户端,缺点是必须使用GMail帐号,有一点是比较方便,不需要写很多代码,但是不是很灵活. 第二种:基于JMail实现,可以很灵活的自己设置各种属性,不需要GMail帐号 在第二种方式的实现之前,看一下JMail对EMail结构的划分: 基于SMTP协议发送EMail,所以客户端必须要知道SMTP的主机. 腾讯邮件的SMTP主机为:s

  • Android HTTP发送请求和接收响应的实例代码

    添加权限 首先要在manifest中加上访问网络的权限: 复制代码 代码如下: <manifest ... > <uses-permission android:name="android.permission.INTERNET" /> ... </manifest> 完整的Manifest文件如下: 复制代码 代码如下: <?xml version="1.0" encoding="utf-8"?>

  • Android发送短信功能代码

    本文实例讲述了Android发送短信功能代码,是进行Android程序开发中非常常见的重要功能.分享给大家供大家参考之用.具体方法如下: // 发送短信 public void sendMsg(){ String content = edtSend.getText().toString(); SmsManager smsManager = SmsManager.getDefault(); List<String> divideContents = smsManager.divideMessag

  • Android开发中怎样调用系统Email发送邮件(多种调用方式)

    我们都知道,在Android中调用其他程序进行相关处理,几乎都是使用的Intent,所以,Email也不例外. 在Android中,调用Email有三种类型的Intent: Intent.ACTION_SENDTO 无附件的发送 Intent.ACTION_SEND 带附件的发送 Intent.ACTION_SEND_MULTIPLE 带有多附件的发送 当然,所谓的调用Email,只是说Email可以接收Intent并做这些事情,可能也有其他的应用程序实现了相关功能,所以在执行的时候,会出现选择

  • android实现蓝牙文件发送的实例代码,支持多种机型

    最近项目上需要实现蓝牙传输apk的一个功能,能够搜索周围的蓝牙手机并分享文件.从需求上讲android手机自带的蓝牙传输模块就可以满足需要了,实现也很简单.不过让人头疼的是,虽然说一般的主流机型都配置有蓝牙模块,但是android机型碎片化太严重,不同android版本手机蓝牙功能也不一样.4.0.3以下版本和以上版本使用的蓝牙包是不同的,分别是"com.android.bluetooth"和"com.mediatek.bluetooth".还有一些厂商对蓝牙模块进

  • Android 后台发送邮件示例 (收集应用异常信息+Demo代码)

    上一次说了如何收集我们已经发布的应用程序的错误信息,方便我们调试完善程序.上次说的收集方法主要是把收集的信息通过Http的post请求把相关的异常信息变成请求参数发送到服务器.这个对做过web开发的人来说,服务端处理是很简单.不过对很多没做个web的人来说却是麻烦事.今天介绍个更简单的方法,我们把异常信息收集后,通过后台发送邮件方法,把相关异常信息发送到我们指定的邮箱里面. 这篇文章是实用性文章,不会涉及太多理论分析.主要是让大家看了以后知道怎么在自己的应用里面添加这个功能. 1.第三方库这次发

  • Android发送GET与POST请求的DEMO详解

    4.0后网络访问必须单独起一个子线程访问,否则无法运行,这里有一个发送请求的工具类GetPostUtil 复制代码 代码如下: public class GetPostUtil{ /**  * 向指定URL发送GET方法的请求  *   * @param url  *            发送请求的URL  * @param params  *            请求参数,请求参数应该是name1=value1&name2=value2的形式.  * @return URL所代表远程资源的

  • Android中Webview打开网页的同时发送HTTP头信息方法

    众所周知,当你点击一个超链接进行跳转时,WebView会自动将当前地址作为Referer(引荐)发给服务器,因此很多服务器端程序通过是否包含referer来控制盗链,所以有些时候,直接输入一个网络地址,可能有问题,那么怎么解决盗链控制问题呢,其实在webview加载时加入一个referer就可以了,如何添加呢? 从Android 2.2 (也就是API 8)开始,WebView新增加了一个接口方法,就是为了便于我们加载网页时又想发送其他的HTTP头信息的. 复制代码 代码如下: public v

随机推荐