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

作者:zeo 时间:2023-04-14 01:02:28 

本文实例讲述了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,彩信,附件
0
投稿

猜你喜欢

  • 浅谈Java中Lambda表达式的相关操作

    2023-09-17 12:00:58
  • Spring Boot实现文件上传下载

    2021-11-22 21:19:36
  • c#中task与thread区别及其使用的方法示例

    2021-08-06 13:59:10
  • 简单阐述一下Java集合的概要

    2023-08-23 19:49:45
  • java swing标准对话框具体实现

    2021-12-30 02:48:50
  • C#实现动态加载dll的方法

    2022-11-26 08:15:17
  • 解决Spring国际化文案占位符失效问题的方法

    2022-10-20 23:42:23
  • Android中RecyclerView 滑动时图片加载的优化

    2021-07-29 09:58:12
  • SpringCloud之熔断器Hystrix的实现

    2021-09-21 01:39:26
  • Java实现单向链表反转

    2023-11-18 01:03:11
  • Springcloud Eureka配置及集群代码实例

    2021-09-19 05:54:52
  • C#位运算以及实例计算详解

    2021-06-03 09:03:20
  • 初识Java环境变量配置及IDEA

    2022-09-20 06:46:41
  • 如何将Mybatis连接到ClickHouse

    2023-11-06 02:35:51
  • python如何调用java类

    2022-03-13 16:11:10
  • AndroidStudio:手势识别

    2022-04-04 00:53:58
  • 简单讲解Java设计模式编程中的单一职责原则

    2022-05-02 03:09:24
  • C#之Expression表达式树实例

    2023-02-24 20:54:31
  • Java并发的CAS原理与ABA问题的讲解

    2023-11-25 12:17:21
  • maven之packaging标签的使用

    2021-10-25 05:14:42
  • asp之家 软件编程 m.aspxhome.com