編輯:關於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程序設計有所幫助。
1.打開手機QQ浏覽器,點擊底欄【菜單】 2.向左滑動,選擇【省流加速】 3.看到【廣告過濾】了嗎,點擊進入 4.在這裡即可選擇是否打開【廣告過濾】
本文講述了Android使用國內鏡像在線更新SDK的方法。分享給大家供大家參考,具體如下:什麼是Android SDK:SDK:(software development
(1)布局文件沒有做更改 (2)在res--menu目錄下的main.xml文件中代碼如下:
然而這個菜單效果只是普通的側拉效果 我們還可以實現抽屜式側滑菜單 就像這樣第一種效果第二種效果第三種效果第四種效果其它代碼都和上篇文章相同,只是在MyHorizontal