Android錄制聲音文件(音頻)并播放
本文實例為大家分享了Android錄制音頻文件的具體代碼,供大家參考,具體內(nèi)容如下
1、這個demo中沒有對多次點擊同一個聲音文件做詳細(xì)處理,偶爾會有崩潰,用的時候需要注意。
2、按住錄音按鈕錄音過程中,只對豎直方向處理了一下,水平方向沒寫;
3、沒有做刪除某個聲音文件的操作,但是測試的時候?qū)崿F(xiàn)了功能,需要用到的話,在MainActivity—>onItemClick中的TODO中有詳細(xì)說明;
4、這只是個demo,如果要在項目中使用,先寫出demo,沒問題了,再引入項目,在寫成demo后,在真機(jī)上運行的時候,如果出現(xiàn)獲取錄音權(quán)限,最好選擇“允許”,如果拒絕,可能會崩潰。
記得打開手機(jī)運行錄音的權(quán)限
先來效果圖:

目錄結(jié)構(gòu):

1、添加權(quán)限:
<uses-permission android:name="android.permission.RECORD_AUDIO"/> <uses-permission android:name="android.permission.WAKE_LOCK"/> <uses-permission android:name="android.permission.MODIFY_AUDIO_SETTINGS"/> <uses-permission android:name="android.permission.VIBRATE"/> <uses-permission android:name="android.permission.WRITE_SETTINGS"/> <uses-permission android:name="android.permission.INTERACT_ACROSS_USERS_FULL"/> <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/> <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"/> <uses-permission android:name="android.permission.CALL_PHONE"/> <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/> <uses-permission android:name="android.permission.INTERNET" /> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" /> <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" />
2、新建MediaRecorderUtils,復(fù)制以下源碼:
package com.chen.voicedemo;
import android.media.MediaRecorder;
import android.os.Handler;
import android.util.Log;
import android.widget.ImageView;
import java.io.File;
/**
* 錄音工具類
*/
public class MediaRecorderUtils {
private static MediaRecorder recorder;
static MediaRecorderUtils mediaRecorderUtils;
static ImageView mimageView;
private String path;
/**
* 獲得單例對象,傳入一個顯示音量大小的imageview對象,如不需要顯示可以傳null
*/
public static MediaRecorderUtils getInstence(ImageView imageView) {
if (mediaRecorderUtils == null) {
mediaRecorderUtils = new MediaRecorderUtils();
}
mimageView = imageView;
return mediaRecorderUtils;
}
/**
* 獲得音頻路徑
*/
public String getPath() {
return path;
}
/**
* 初始化
*/
private void init() {
recorder = new MediaRecorder();// new出MediaRecorder對象
recorder.setAudioSource(MediaRecorder.AudioSource.MIC);
// 設(shè)置MediaRecorder的音頻源為麥克風(fēng)
recorder.setOutputFormat(MediaRecorder.OutputFormat.RAW_AMR);
// 設(shè)置MediaRecorder錄制的音頻格式
recorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
// 設(shè)置MediaRecorder錄制音頻的編碼為amr.
File file = new File(Utils.IMAGE_SDCARD_MADER);
if (!file.exists()) {
file.mkdirs();
}
path = Utils.IMAGE_SDCARD_MADER + Utils.getVoiceFileName() + "stock.amr";
recorder.setOutputFile(path);
// 設(shè)置錄制好的音頻文件保存路徑
try {
recorder.prepare();// 準(zhǔn)備錄制
} catch (Exception e) {
e.printStackTrace();
}
}
/**
* 開始錄音
*/
public void MediaRecorderStart() {
init();
try {
recorder.start();
flag = true;
if (mimageView != null) {
updateMicStatus();
}
} catch (Exception e) {
e.printStackTrace();
Log.e("chen", "錄制失敗");
}
}
/**
* 停止錄音
*/
public void MediaRecorderStop() {
try {
recorder.stop();
recorder.release(); //釋放資源
flag = false;
mimageView = null;
recorder = null;
} catch (Exception e) {
e.toString();
}
}
/**
* 刪除已錄制的音頻
*/
public void MediaRecorderDelete() {
File file = new File(path);
if (file.isFile()) {
file.delete();
}
file.exists();
}
;
private final Handler mHandler = new Handler();
private Runnable mUpdateMicStatusTimer = new Runnable() {
public void run() {
updateMicStatus();
}
};
private int BASE = 1;
private int SPACE = 1000;// 間隔取樣時間
private boolean flag = true;
/**
* 更新話筒狀態(tài)
*/
private void updateMicStatus() {
if (recorder != null) {
double ratio = (double) recorder.getMaxAmplitude() / BASE;
double db = 0;// 分貝
if (ratio > 1) {
db = 20 * Math.log10(ratio);
}
int i = (int) db / 10;
switch (i) {
case 1:
mimageView.setImageResource(R.drawable.rc_ic_volume_1);
break;
case 2:
mimageView.setImageResource(R.drawable.rc_ic_volume_2);
break;
case 3:
mimageView.setImageResource(R.drawable.rc_ic_volume_3);
break;
case 4:
mimageView.setImageResource(R.drawable.rc_ic_volume_4);
break;
case 5:
mimageView.setImageResource(R.drawable.rc_ic_volume_5);
break;
case 6:
mimageView.setImageResource(R.drawable.rc_ic_volume_6);
break;
case 7:
mimageView.setImageResource(R.drawable.rc_ic_volume_7);
break;
case 8:
mimageView.setImageResource(R.drawable.rc_ic_volume_8);
break;
}
if (flag) {
mHandler.postDelayed(mUpdateMicStatusTimer, SPACE);
}
}
}
}
3、創(chuàng)建MyChronometer,復(fù)制以下代碼
package com.chen.voicedemo;
import android.annotation.SuppressLint;
import android.content.Context;
import android.os.Handler;
import android.os.Message;
import android.os.SystemClock;
import android.util.AttributeSet;
import android.view.accessibility.AccessibilityEvent;
import android.view.accessibility.AccessibilityNodeInfo;
import android.widget.TextView;
public class MyChronometer extends TextView {
private static final String TAG = "MyChronometer";
/**
* A callback that notifies when the MyChronometer has incremented on its
* own.
*/
public interface OnMyChronometerTickListener {
/**
* Notification that the MyChronometer has changed.
*/
void onMyChronometerTick(int time);
}
public interface OnMyChronometerTimeListener {
/**
* Notification that the MyChronometer has changed.
*/
void OnMyChronometerTimeListener(int time);
}
private OnMyChronometerTimeListener OnMyChronometerTimeListener;
private long mBase;
private boolean mVisible;
private boolean mStarted;
private boolean mRunning;
private OnMyChronometerTickListener mOnMyChronometerTickListener;
private long now_time;
private static final int TICK_WHAT = 2;
/**
* Initialize this MyChronometer object. Sets the base to the current time.
*/
public MyChronometer(Context context) {
this(context, null, 0);
}
/**
* Initialize with standard view layout information. Sets the base to the
* current time.
*/
public MyChronometer(Context context, AttributeSet attrs) {
this(context, attrs, 0);
}
/**
* Initialize with standard view layout information and style. Sets the base
* to the current time.
*/
public MyChronometer(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
init();
}
private void init() {
mBase = SystemClock.elapsedRealtime();
updateText(mBase);
}
/**
* Set the time that the count-up timer is in reference to.
*
* @param base Use the {@link SystemClock#elapsedRealtime} time base.
*/
public void setBase(long base) {
mBase = base;
updateText(SystemClock.elapsedRealtime());
}
/**
* Sets the listener to be called when the MyChronometer changes.
*
* @param listener The listener.
*/
public void setOnMyChronometerTickListener(OnMyChronometerTickListener listener) {
mOnMyChronometerTickListener = listener;
}
public void setOnMyChronometerTimeListener(OnMyChronometerTimeListener listener) {
OnMyChronometerTimeListener = listener;
}
/**
* Start counting up. This does not affect the base as set from
* {@link #setBase}, just the view display.
* <p/>
* MyChronometer works by regularly scheduling messages to the handler, even
* when the Widget is not visible. To make sure resource leaks do not occur,
* the user should make sure that each start() call has a reciprocal call to
* {@link #stop}.
*/
public void start() {
mStarted = true;
updateRunning();
}
/**
* Stop counting up. This does not affect the base as set from
* {@link #setBase}, just the view display.
* <p/>
* This stops the messages to the handler, effectively releasing resources
* that would be held as the MyChronometer is running, via {@link #start}.
*/
public void stop() {
mStarted = false;
updateRunning();
now_time /= 10;
if (OnMyChronometerTimeListener != null) {
OnMyChronometerTimeListener.OnMyChronometerTimeListener((int) now_time);
}
}
@Override
protected void onDetachedFromWindow() {
super.onDetachedFromWindow();
mVisible = false;
updateRunning();
}
@Override
protected void onWindowVisibilityChanged(int visibility) {
super.onWindowVisibilityChanged(visibility);
mVisible = visibility == VISIBLE;
updateRunning();
}
private synchronized void updateText(long now) {
long seconds = now - mBase;
seconds /= 10;
now_time = seconds;
int time_m = (int) (seconds / 100);
if (mOnMyChronometerTickListener != null) {
mOnMyChronometerTickListener.onMyChronometerTick(time_m);
}
int time_s = (int) (seconds % 100);
setText(time_m + "");
}
private void updateRunning() {
boolean running = mVisible && mStarted;
if (running != mRunning) {
if (running) {
updateText(SystemClock.elapsedRealtime());
mHandler.sendMessageDelayed(Message.obtain(mHandler, TICK_WHAT), 1000);
} else {
mHandler.removeMessages(TICK_WHAT);
}
mRunning = running;
}
}
private Handler mHandler = new Handler() {
public void handleMessage(Message m) {
if (mRunning) {
updateText(SystemClock.elapsedRealtime());
sendMessageDelayed(Message.obtain(this, TICK_WHAT), 1000);
}
}
};
@SuppressLint("NewApi")
@Override
public void onInitializeAccessibilityEvent(AccessibilityEvent event) {
super.onInitializeAccessibilityEvent(event);
event.setClassName(MyChronometer.class.getName());
}
@SuppressLint("NewApi")
@Override
public void onInitializeAccessibilityNodeInfo(AccessibilityNodeInfo info) {
super.onInitializeAccessibilityNodeInfo(info);
info.setClassName(MyChronometer.class.getName());
}
}
4、創(chuàng)建工具類
package com.chen.voicedemo;
import android.Manifest;
import android.content.Context;
import android.content.pm.PackageManager;
import android.os.Environment;
import android.support.v4.content.ContextCompat;
import android.widget.Toast;
import java.io.File;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
/**
* 工具
*/
public class Utils {
/**
* SD卡下語音目錄
*/
public static final String IMAGE_SDCARD_MADER = Environment
.getExternalStorageDirectory()
+ "/chen/voice/";
/**
* 檢查錄音權(quán)限6.0
*/
public static boolean checkVoice(Context context) {
try {
if (ContextCompat.checkSelfPermission(context, Manifest.permission.RECORD_AUDIO) != PackageManager.PERMISSION_GRANTED) {
return false;
} else {
return true;
}
} catch (Exception e) {
return true;
}
}
private static Toast toast;
/**
* 單例吐司
*/
public static void showToast(Context context, String msg) {
if (toast == null) {
toast = Toast.makeText(context, msg, Toast.LENGTH_SHORT);
}
toast.setText(msg);
toast.show();
}
/**
* 獲取指定文件夾下的所有文件路徑
*
* @param root 指定文件夾路徑
* @return 指定文件夾下的所有文件
*/
public static ArrayList<String> getVideoFiles(String root) {
if (root == null || root == "")
return null;
ArrayList<String> list = new ArrayList<>();
File file = new File(root);
File[] fileList = file.listFiles();
for (File f : fileList) {
list.add(f.getPath());
}
return list;
}
/**
* 獲取聲音文件名字
*
* @return 假如當(dāng)前錄制聲音時間是2016年4月29號14點30分30秒。得到的文件名字就是20160429143030.這樣保證文件名的唯一性
*/
public static String getVoiceFileName() {
long getNowTimeLong = System.currentTimeMillis();
SimpleDateFormat time = new SimpleDateFormat("yyyyMMddHHmmss");
String result = time.format(getNowTimeLong);
return result;
}
}
5、MainActivity
package com.chen.voicedemo;
import android.app.Activity;
import android.graphics.drawable.AnimationDrawable;
import android.graphics.drawable.ColorDrawable;
import android.media.MediaPlayer;
import android.os.Bundle;
import android.os.SystemClock;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewGroup;
import android.view.Window;
import android.view.WindowManager;
import android.widget.AdapterView;
import android.widget.BaseAdapter;
import android.widget.ImageView;
import android.widget.ListView;
import android.widget.PopupWindow;
import android.widget.TextView;
import java.io.File;
import java.util.ArrayList;
import java.util.List;
public class MainActivity extends Activity implements View.OnTouchListener, AdapterView.OnItemClickListener {
/**
* 開始錄音按鈕
*/
private TextView voice;
/**
* 用于定位。使錄音時展示的popupwindow,展示在該控件 的下面
*/
private TextView voice_popup;
/**
* 展示指定文件夾下所有錄制的聲音文件
*/
private TextView show_voice_list;
/**
* 展示目標(biāo)文件夾下,所有已錄制的聲音路徑
*/
private ListView show_voices_listview;
private List<String> voiceList;
/**
* 停止播放聲音
*/
private TextView stop_show_voice;
/**
* 播放聲音時,動的圖片
*/
private ImageView voice_anim;
/**
* 系統(tǒng)播放器
*/
private MediaPlayer mediaPlayer;
private Boolean flag = true;
private float int_x = 0;
private float int_y = 0;
/**
* 用于限制最大錄音時常。單位是秒。意義是:最大錄60秒的音頻,到了60秒的是,自動停止
*/
private int maxRecordTime = 60;
/**
* 用于顯示頻繁操作時間間隔。單位是毫秒。意義是:500毫秒內(nèi)再次操作,就算是頻頻操作,做相應(yīng)處理
*/
private int oftenOperationTime = 500;
private MyAdapter myAdapter;
private AnimationDrawable animation;
/**
* 錄音popup
*/
private PopupWindow voice_popupWindow;
/**
* 錄音時聲音變化
*/
private ImageView voice_shengyin;
/**
* 錄音計時器
*/
private MyChronometer mychronometer;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
requestWindowFeature(Window.FEATURE_NO_TITLE);
setContentView(R.layout.activity_main);
voiceList = new ArrayList<String>();
voice = (TextView) findViewById(R.id.voice);
voice_popup = (TextView) findViewById(R.id.voice_popup);
voice_anim = (ImageView) findViewById(R.id.voice_anim);
voice_anim.setImageResource(R.drawable.lcs_voice_receive);
show_voice_list = (TextView) findViewById(R.id.show_voice_list);
show_voice_list.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
voiceList = Utils.getVideoFiles(Utils.IMAGE_SDCARD_MADER);
if (voiceList.size()>0){
myAdapter.notifyDataSetChanged();
}else{
Utils.showToast(MainActivity.this, "沒有文件");
}
}
});
show_voices_listview = (ListView) findViewById(R.id.show_voices);
show_voices_listview.setOnItemClickListener(this);
myAdapter = new MyAdapter();
stop_show_voice = (TextView) findViewById(R.id.stop_show_voice);
/**
* 停止播放的監(jiān)聽器
*/
stop_show_voice.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Log.e("chen", "點擊了停止播放按鈕");
if (mediaPlayer != null) {
if (mediaPlayer.isPlaying()) {
mediaPlayer.release();// 釋放資源
}
mediaPlayer = null;
}
if (animation != null && animation.isRunning()) {
animation.stop();
}
voice_anim.setImageResource(R.drawable.lcs_voice_receive);
}
});
show_voices_listview.setAdapter(myAdapter);
voice.setOnTouchListener(this);
}
/**
* 聲音文件列表的item點擊事件,播放對應(yīng)聲音文件
*
* @param parent
* @param view
* @param position
* @param id
*/
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
//TODO 以下4行,是用來做測試,點擊item,手機(jī)SD卡上對應(yīng)路徑下的聲音文件就會被刪除。如果錄制聲音失敗,或者不滿足條件,可以把以下4行寫成一個工具方法調(diào)用,刪除不滿意的文件。這里不做詳細(xì)演示
//File f_delete=new File(voiceList.get(position));
//f_delete.delete();
//voiceList.remove(voiceList.get(position));
//myAdapter.notifyDataSetChanged();
//TODO 以上4行,是用來做測試,點擊item,手機(jī)SD卡上對應(yīng)路徑下的聲音文件就會被刪除。
try {
mediaPlayer = new MediaPlayer();
/**
* 播放過程中展示的動畫
*/
mediaPlayer.setOnPreparedListener(new MediaPlayer.OnPreparedListener() {
@Override
public void onPrepared(MediaPlayer mp) {
if (mp != null) {
mp.start();
voice_anim.setImageResource(R.drawable.voice_anim);
}
}
});
/**
* 播放完成監(jiān)聽
*/
mediaPlayer.setOnCompletionListener(new MediaPlayer.OnCompletionListener() {
@Override
public void onCompletion(MediaPlayer mp) {
if (mp.isPlaying()) {
mp.release();// 釋放資源
}
animation = (AnimationDrawable) voice_anim.getDrawable();
if (animation != null && animation.isRunning()) {
animation.stop();
}
voice_anim.setImageResource(R.drawable.lcs_voice_receive);
}
});
mediaPlayer.setDataSource(voiceList.get(position));
// 緩沖
mediaPlayer.prepare();
} catch (Exception e) {
Utils.showToast(MainActivity.this, "語音異常,加載失敗");
}
}
/**
* 展示聲音列表的adapter
*/
class MyAdapter extends BaseAdapter {
@Override
public int getCount() {
return voiceList.size() == 0 ? 0 : voiceList.size();
}
@Override
public Object getItem(int position) {
return null;
}
@Override
public long getItemId(int position) {
return 0;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
TextView tv = new TextView(MainActivity.this);
tv.setText(voiceList.get(position));
tv.setTextSize(20);
return tv;
}
}
/**
* 開始錄制按鈕的onTouch事件
*
* @param v
* @param event
* @return
*/
@Override
public boolean onTouch(View v, MotionEvent event) {
if (v.getId() == R.id.voice) {
//檢查權(quán)限
if (!Utils.checkVoice(this)) {
if (event.getAction() == MotionEvent.ACTION_DOWN) {
Utils.showToast(this, "錄音權(quán)限未打開,請打開錄音權(quán)限!");
}
return true;
}
//避免短時間里頻繁操作
if (!getTimeTF(SystemClock.elapsedRealtime()) && event.getAction() == MotionEvent.ACTION_DOWN) {
Utils.showToast(this, "操作過于頻繁");
return true;
}
if (event.getAction() == MotionEvent.ACTION_DOWN) {
setTime(SystemClock.elapsedRealtime());
}
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
int_x = event.getRawX();
int_y = event.getRawY();
VoicePopupWindow();
mychronometer.setBase(SystemClock.elapsedRealtime());
mychronometer.start();
MediaRecorderUtils.getInstence(voice_shengyin).MediaRecorderStart();
flag = true;
mychronometer.setOnMyChronometerTickListener(new MyChronometer.OnMyChronometerTickListener() {
@Override
public void onMyChronometerTick(int time) {
if (time == maxRecordTime || time > maxRecordTime) {
mychronometer.setText("60");
setVoiceToUp();
}
}
});
break;
case MotionEvent.ACTION_MOVE:
if (flag) {
if (Math.abs(int_y) - Math.abs(event.getRawY()) > 100.0 && flag) {
voice_popupWindow.dismiss();
mychronometer.stop();
MediaRecorderUtils.getInstence(voice_shengyin).MediaRecorderStop();
MediaRecorderUtils.getInstence(voice_shengyin).MediaRecorderDelete();
flag = false;
}
}
break;
case MotionEvent.ACTION_CANCEL:
if (flag) {
voice_popupWindow.dismiss();
mychronometer.stop();
MediaRecorderUtils.getInstence(voice_shengyin).MediaRecorderStop();
}
break;
case MotionEvent.ACTION_UP:
if (flag) {
setVoiceToUp();
}
break;
}
return true;
}
return false;
}
private long base_time = 0;
private void setTime(long time) {
base_time = time;
}
private boolean getTimeTF(long time) {
int data = (int) (time - base_time) / oftenOperationTime;
if (data > 1) {
return true;
} else {
return false;
}
}
/**
* 聲音popupwindow
*/
public void VoicePopupWindow() {
View view = LayoutInflater.from(this).inflate(R.layout.voice_popupwindow, null);
voice_popupWindow = new PopupWindow(this);
voice_popupWindow.setWidth(WindowManager.LayoutParams.MATCH_PARENT);
voice_popupWindow.setHeight(WindowManager.LayoutParams.MATCH_PARENT);
voice_shengyin = (ImageView) view.findViewById(R.id.voice_shengyin);
mychronometer = (MyChronometer) view.findViewById(R.id.mychronometer);
voice_popupWindow.setContentView(view);
voice_popupWindow.setFocusable(true);
ColorDrawable dw = new ColorDrawable(0x00000000);
voice_popupWindow.setBackgroundDrawable(dw);
voice_popupWindow.showAsDropDown(voice_popup);
}
private void setVoiceToUp() {
flag = false;
voice_popupWindow.dismiss();
mychronometer.stop();
MediaRecorderUtils.getInstence(voice_shengyin).MediaRecorderStop();
int time = Integer.parseInt(mychronometer.getText().toString());
if (time != 0) {
File file = new File(MediaRecorderUtils.getInstence(voice_shengyin).getPath());
if (file.length() > 0) {
voiceList = Utils.getVideoFiles(Utils.IMAGE_SDCARD_MADER);
myAdapter.notifyDataSetChanged();
} else {
Utils.showToast(this, "錄音失敗,請檢查權(quán)限");
}
} else {
Utils.showToast(this, "錄音時間太短");
}
}
}
6、activity_main布局
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" > <TextView android:id="@+id/voice_popup" android:layout_width="match_parent" android:layout_height="1dip"/> <ListView android:id="@+id/show_voices" android:layout_width="match_parent" android:layout_height="0dp" android:layout_weight="1"/> <RelativeLayout android:layout_width="match_parent" android:layout_height="wrap_content" android:orientation="horizontal"> <ImageView android:id="@+id/voice_anim" android:layout_width="60dp" android:layout_height="30dp" android:layout_centerVertical="true" android:layout_marginLeft="30dp" android:background="#00ff00"/> <TextView android:id="@+id/stop_show_voice" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_alignParentRight="true" android:layout_centerVertical="true" android:layout_gravity="center_horizontal" android:layout_marginBottom="20dp" android:layout_marginRight="20dp" android:background="#00ff00" android:padding="10dp" android:text="停止播放" android:textColor="#000000" android:textSize="20sp" /> <TextView android:id="@+id/show_voice_list" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_centerVertical="true" android:layout_gravity="center_horizontal" android:layout_marginBottom="20dp" android:layout_marginRight="20dp" android:layout_toLeftOf="@id/stop_show_voice" android:background="#00ff00" android:padding="10dp" android:text="列表" android:textColor="#000000" android:textSize="20sp" /> </RelativeLayout> <TextView android:id="@+id/voice" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_gravity="center_horizontal" android:background="#00ff00" android:padding="10dp" android:text="開始錄音" android:textColor="#000000" android:textSize="25sp"/> </LinearLayout>
7、voice_popupwindow布局代碼:錄音的時候,會出現(xiàn)以下圖片中的popupwindow

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent">
<LinearLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:background="@android:color/black"
android:orientation="vertical"
android:paddingBottom="40dip"
android:paddingLeft="60dip"
android:paddingRight="60dip"
android:paddingTop="40dip">
<ImageView
android:id="@+id/voice_shengyin"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center_horizontal"
android:src="@drawable/rc_ic_volume_1"/>
<com.chen.voicedemo.MyChronometer
android:id="@+id/mychronometer"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerHorizontal="true"
android:layout_gravity="center_horizontal"
android:textColor="@android:color/white"/>
</LinearLayout>
</RelativeLayout>
8、還有一個動畫布局,播放聲音的時候,有個動畫效果

<?xml version="1.0" encoding="utf-8"?>
<animation-list xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/volume_animation"
android:oneshot="false" >
<item
android:drawable="@drawable/rc_ic_voice_receive_play1"
android:duration="100"/>
<item
android:drawable="@drawable/rc_ic_voice_receive_play2"
android:duration="200"/>
<item
android:drawable="@drawable/rc_ic_voice_receive_play3"
android:duration="300"/>
</animation-list>
附錄:用到的圖片資源說明:如果手上沒有這樣的圖片,可以隨便用其他圖片代替,有效果,就算成功


以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Android MediaPlayer 音頻倍速播放 調(diào)整播放速度問題
- Android MediaPlayer 播放音頻的方式
- 詳解Android開發(fā)錄音和播放音頻的步驟(動態(tài)獲取權(quán)限)
- Android使用SoundPool實現(xiàn)播放音頻
- Android自定義View實現(xiàn)音頻播放圓形進(jìn)度條
- Android多媒體應(yīng)用使用SoundPool播放音頻
- Android多媒體應(yīng)用使用MediaPlayer播放音頻
- Android開發(fā)之MediaPlayer多媒體(音頻,視頻)播放工具類
- Android編程實現(xiàn)播放音頻的方法示例
- Android音頻處理之通過AudioRecord去保存PCM文件進(jìn)行錄制,播放,停止,刪除功能
- Android App中使用AudioManager類來編寫音頻播放器
- Android提高之MediaPlayer播放網(wǎng)絡(luò)音頻的實現(xiàn)方法
- android實現(xiàn)小音頻頻繁播放
相關(guān)文章
Android實現(xiàn)漸變圓環(huán)、圓形進(jìn)度條效果
這篇文章主要為大家詳細(xì)介紹了Android實現(xiàn)漸變圓環(huán)、圓形進(jìn)度條效果,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-10-10
Android入門之IntentService的使用教程詳解
IntentService的生命周期中有一個非常好的方法-onHandleIntent方法,它是一個abstract方法,開發(fā)者在實現(xiàn)IntentService時可以覆蓋它來處理“長事務(wù)”。本文就來聊聊IntentService的使用,需要的可以參考一下2022-12-12
Android EdText編輯框禁止輸入表情符號(使用正則表達(dá)式)
這篇文章主要介紹了Android EdText編輯框禁止輸入表情符號使用正則表達(dá)式,需要的朋友可以參考下2017-06-06
Android?無障礙服務(wù)?performAction?調(diào)用過程分析
這篇文章主要介紹了Android?無障礙服務(wù)?performAction?調(diào)用過程分析,無障礙服務(wù)可以模擬一些用戶操作,無障礙可以處理的對象,通過類?AccessibilityNodeInfo?表示,通過無障礙服務(wù),可以通過它的performAction方法來觸發(fā)一些action2022-06-06
Android 中menu同時顯示圖標(biāo)和文字的實現(xiàn)
這篇文章主要介紹了Android 中menu同時顯示圖標(biāo)和文字的實現(xiàn)的相關(guān)資料,希望通過本文能幫助到大家實現(xiàn)這樣的功能,需要的朋友可以參考下2017-10-10
Android性能優(yōu)化之利用強(qiáng)大的LeakCanary檢測內(nèi)存泄漏及解決辦法
本篇文章主要介紹了Android性能優(yōu)化之利用LeakCanary檢測內(nèi)存泄漏及解決辦法,有興趣的同學(xué)可以了解一下。2016-11-11

