Android实现多媒体录音笔

 更新时间:2016年01月15日 10:57:05   投稿:lijiao  
这篇文章主要介绍了Android实现多媒体录音笔的相关资料,以及在实现过程中遇到问题的解决方法,感兴趣的小伙伴们可以参考一下

记事本涉及到的仅仅是对string 的存储,而且在读取上并不存在什么难点,直接用textview显示便可以了。需要做的主要是使用SQLite对数据进行一个整理。
而录音笔需要考虑的就相对较多了:比如录音时中断,录音时用户点击播放按钮;未录音,用户点击停止按钮;在录音或者播放时关闭activity;listview的item中需要为button设置不同的点击效果等等。

为了便于新手学习,在此还是罗列一下涉及的主要知识点:

  • 1、Baseadapter
  • 2、JAVA的file
  • 3、MediaRecorder
  • 4、较多的AlertDialog
  • 5、MediaPlayer

遇到的问题:
在listview item中的button控件可以获得焦点时,直接为listview设置item长按事件的监听。出现了listview的item长按事件无效的情况。

解决方法:
直接在Baseadapter中对该item的布局进行长按事件的监听(在笔者demo中是linearlayout),也就是说对item中button的父布局进行长按事件的监听。

效果:

   

MainActivity:

package com.example.recorder; 
 
import android.app.Activity; 
import android.app.AlertDialog; 
import android.app.AlertDialog.Builder; 
import android.content.DialogInterface; 
import android.media.MediaPlayer; 
import android.media.MediaRecorder; 
import android.os.Bundle; 
import android.os.Environment; 
import android.view.LayoutInflater; 
import android.view.View; 
import android.view.View.OnClickListener; 
import android.view.ViewGroup; 
import android.widget.AdapterView; 
import android.widget.BaseAdapter; 
import android.widget.Button; 
import android.widget.EditText; 
import android.widget.LinearLayout; 
import android.widget.ListView; 
import android.widget.TextView; 
import android.widget.Toast; 
 
import java.io.File; 
import java.io.IOException; 
 
public class MainActivity extends Activity implements OnClickListener { 
 
  private Button start; 
  private Button stop; 
  private ListView listView; 
  ShowRecorderAdpter showRecord; 
 
  // 录音文件播放 
  // 录音 
  // 音频文件保存地址 
  private MediaPlayer myPlayer; 
  private MediaRecorder myRecorder = null; 
  private String path; 
  private File saveFilePath; 
  // 所录音的文件名 
  String[] listFile = null; 
 
 
  @Override 
  protected void onCreate(Bundle savedInstanceState) { 
    super.onCreate(savedInstanceState); 
    setContentView(R.layout.activity_main); 
 
    //初始化控件 
    InitView(); 
 
  } 
 
  private void InitView() { 
    start = (Button) findViewById(R.id.start); 
    stop = (Button) findViewById(R.id.stop); 
    listView = (ListView) findViewById(R.id.list); 
 
    myPlayer = new MediaPlayer(); 
    showRecord = new ShowRecorderAdpter(); 
 
    //如果手机有sd卡 
    if (Environment.getExternalStorageState().equals( 
        Environment.MEDIA_MOUNTED)) { 
      try { 
        path = Environment.getExternalStorageDirectory() 
            .getCanonicalPath().toString() 
            + "/MyRecorders"; 
        File files = new File(path); 
        if (!files.exists()) { 
          //如果有没有文件夹就创建文件夹 
          files.mkdir(); 
        } 
        listFile = files.list(); 
      } catch (IOException e) { 
        e.printStackTrace(); 
      } 
    } 
 
    start.setOnClickListener(this); 
    stop.setOnClickListener(this); 
    listView.setAdapter(showRecord); 
 
 
  } 
 
 
  //由于在item中涉及到了控件的点击效果,所以采用BaseAdapter 
  class ShowRecorderAdpter extends BaseAdapter { 
 
    @Override 
    public int getCount() { 
      return listFile.length; 
    } 
 
    @Override 
    public Object getItem(int arg0) { 
      return arg0; 
    } 
 
    @Override 
    public long getItemId(int arg0) { 
      return arg0; 
 
    } 
 
    @Override 
    public View getView(final int postion, View arg1, ViewGroup arg2) { 
      View views = LayoutInflater.from(MainActivity.this).inflate( 
          R.layout.list_item, null); 
      LinearLayout parent = (LinearLayout) views.findViewById(R.id.list_parent); 
      TextView filename = (TextView) views.findViewById(R.id.show_file_name); 
      Button plays = (Button) views.findViewById(R.id.bt_list_play); 
      Button stop = (Button) views.findViewById(R.id.bt_list_stop); 
 
      //在textview中显示的时候把“.amr”去掉 
      filename.setText(listFile[postion].substring(0, listFile[postion].length() - 4)); 
      parent.setOnLongClickListener(new View.OnLongClickListener() { 
        @Override 
        public boolean onLongClick(View view) { 
          AlertDialog aler = new AlertDialog.Builder(MainActivity.this) 
              .setTitle("确定删除该录音?") 
              .setPositiveButton("确定", new DialogInterface.OnClickListener() { 
                @Override 
                public void onClick(DialogInterface dialog 
                    , int which) { 
                  File file = new File(path + "/" + listFile[postion]); 
                  file.delete(); 
                  // 在删除文件后刷新文件名列表 
                  File files = new File(path); 
                  listFile = files.list(); 
 
                  // 当文件被删除刷新ListView 
                  showRecord.notifyDataSetChanged(); 
                } 
              }) 
              .setNegativeButton("取消", null) 
              .create(); 
          //设置不允许点击提示框之外的区域 
          aler.setCanceledOnTouchOutside(false); 
          aler.show(); 
          return false; 
        } 
      }); 
      // 播放录音 
      plays.setOnClickListener(new OnClickListener() { 
        @Override 
        public void onClick(View arg0) { 
          //确认不是在录音的过程中播放 
          if (myRecorder == null) { 
            try { 
              myPlayer.reset(); 
              myPlayer.setDataSource(path + "/" + listFile[postion]); 
              if (!myPlayer.isPlaying()) { 
                myPlayer.prepare(); 
                myPlayer.start(); 
              } else { 
                myPlayer.pause(); 
              } 
 
            } catch (IOException e) { 
              e.printStackTrace(); 
            } 
          } else { 
            Toast.makeText(MainActivity.this, "请不要再录音的过程中播放!", Toast.LENGTH_SHORT).show(); 
          } 
        } 
      }); 
      // 停止播放 
      stop.setOnClickListener(new OnClickListener() { 
 
        @Override 
        public void onClick(View arg0) { 
          if (myRecorder == null && myPlayer.isPlaying()) { 
            myPlayer.stop(); 
          } 
        } 
      }); 
      return views; 
    } 
 
  } 
 
  @Override 
  public void onClick(View v) { 
    switch (v.getId()) { 
      case R.id.start: 
        final EditText filename = new EditText(this); 
        AlertDialog aler = new Builder(this) 
            .setTitle("请输入要保存的文件名") 
            .setView(filename) 
            .setPositiveButton("确定", 
                new DialogInterface.OnClickListener() { 
                  @Override 
                  public void onClick(DialogInterface dialog, 
                            int which) { 
                    String text = filename.getText().toString(); 
                    //如果文件名为空则跳出提示信息 
                    if (text.equals("")) { 
                      Toast.makeText(MainActivity.this, 
                          "请不要输入空的文件名!", Toast.LENGTH_SHORT).show(); 
                    } else { 
                      //开启录音 
                      RecorderStart(text); 
 
                      start.setText("正在录音中。。"); 
                      start.setEnabled(false); 
                      stop.setEnabled(true); 
                      // 在增添文件后刷新文件名列表 
                      File files = new File(path); 
                      listFile = files.list(); 
                      // 当文件增加刷新ListView 
                      showRecord.notifyDataSetChanged(); 
                    } 
                  } 
                }) 
            .setNegativeButton("取消",null) 
            .create(); 
        //设置不允许点击提示框之外的区域 
        aler.setCanceledOnTouchOutside(false); 
        aler.show(); 
        break; 
      case R.id.stop: 
        myRecorder.stop(); 
        myRecorder.release(); 
        myRecorder = null; 
        // 判断是否保存 如果不保存则删除 
        aler = new AlertDialog.Builder(this) 
            .setTitle("是否保存该录音") 
            .setPositiveButton("确定", null) 
            .setNegativeButton("取消", 
                new DialogInterface.OnClickListener() { 
                  @Override 
                  public void onClick(DialogInterface dialog, 
                            int which) { 
                    saveFilePath.delete(); 
                    // 在删除文件后刷新文件名列表 
                    File files = new File(path); 
                    listFile = files.list(); 
 
                    // 当文件被删除刷新ListView 
                    showRecord.notifyDataSetChanged(); 
                  } 
                }).create(); 
        //设置不允许点击提示框之外的区域 
        aler.setCanceledOnTouchOutside(false); 
        aler.show(); 
 
        start.setText("录音"); 
        start.setEnabled(true); 
        stop.setEnabled(false); 
      default: 
        break; 
    } 
 
  } 
 
  private void RecorderStart(String text) { 
    try { 
      myRecorder = new MediaRecorder(); 
      // 从麦克风源进行录音 
      myRecorder.setAudioSource(MediaRecorder.AudioSource.DEFAULT); 
      // 设置输出格式 
      myRecorder.setOutputFormat(MediaRecorder.OutputFormat.DEFAULT); 
      // 设置编码格式 
      myRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.DEFAULT); 
 
      String paths = path + "/" + text + ".amr"; 
      saveFilePath = new File(paths); 
      myRecorder.setOutputFile(saveFilePath.getAbsolutePath()); 
      myRecorder.prepare(); 
      // 开始录音 
      myRecorder.start(); 
    } catch (Exception e) { 
      e.printStackTrace(); 
    } 
  } 
 
 
  @Override 
  protected void onDestroy() { 
    super.onDestroy(); 
    // 如果myPlayer正在播放,那么就停止播放,并且释放资源 
    if (myPlayer.isPlaying()) { 
      myPlayer.stop(); 
      myPlayer.release(); 
    } 
    //如果myRecorder有内容(代表正在录音),那么就直接释放资源 
    if(myRecorder!=null) { 
      myRecorder.release(); 
      myPlayer.release(); 
    } 
  } 
 
}  

activity_main:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" 
  xmlns:tools="http://schemas.android.com/tools" 
  android:layout_width="match_parent" 
  android:layout_height="match_parent" 
  android:orientation="vertical" 
  tools:context=".MainActivity"> 
 
 
  <TextView 
    android:layout_width="match_parent" 
    android:layout_height="wrap_content" 
    android:background="#000" 
    android:padding="13dp" 
    android:text="语音笔" 
    android:textColor="#fff" 
    android:textSize="22sp" 
    android:textStyle="bold" /> 
 
  <ListView 
    android:id="@+id/list" 
    android:layout_width="match_parent" 
    android:layout_height="0dp" 
    android:layout_weight="1" 
    android:padding="10dp" 
    ></ListView> 
 
  <LinearLayout 
    android:id="@+id/li1" 
    android:padding="10dp" 
    android:layout_width="match_parent" 
    android:layout_height="wrap_content" 
    android:orientation="horizontal"> 
 
    <Button 
      android:id="@+id/start" 
      android:layout_width="0dp" 
      android:layout_height="wrap_content" 
      android:layout_weight="1" 
      android:textSize="20sp" 
      android:text="开始录音" /> 
 
    <Button 
      android:id="@+id/stop" 
      android:layout_width="0dp" 
      android:layout_height="wrap_content" 
      android:layout_weight="1" 
      android:enabled="false" 
      android:textSize="20sp" 
      android:text="停止录音" /> 
  </LinearLayout> 
 
 
</LinearLayout> 

list_item:

<?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:padding="10dp" 
  android:id="@+id/list_parent" 
  android:orientation="horizontal"> 
 
  <TextView 
    android:id="@+id/show_file_name" 
    android:layout_width="0dp" 
    android:layout_weight="1" 
    android:layout_height="wrap_content" 
    android:text="文件名" 
    android:textColor="#000" 
    android:textSize="20sp" 
    /> 
 
  <Button 
    android:id="@+id/bt_list_play" 
    android:layout_width="wrap_content" 
    android:layout_height="wrap_content" 
    android:textSize="18sp" 
    android:text="播放" /> 
 
  <Button 
    android:id="@+id/bt_list_stop" 
    android:layout_width="wrap_content" 
    android:layout_height="wrap_content" 
    android:textSize="18sp" 
    android:text="停止" /> 
 
</LinearLayout>  

以上就是本文的全部内容,希望对大家实现Android软件编程有所帮助。

相关文章

  • Android编程监听APK安装与删除等过程的方法

    Android编程监听APK安装与删除等过程的方法

    这篇文章主要介绍了Android编程监听APK安装与删除等过程的方法,涉及Android事件监听、权限控制、广播操作等相关实现技巧,需要的朋友可以参考下
    2017-10-10
  • Android状态栏的适配汇总

    Android状态栏的适配汇总

    这篇文章主要给大家介绍了关于Android状态栏适配的相关资料,文中通过示例代码介绍的非常详细,对各位Android开发者们具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-05-05
  • Android TabHost选项卡标签图标始终不出现的解决方法

    Android TabHost选项卡标签图标始终不出现的解决方法

    这篇文章主要介绍了Android TabHost选项卡标签图标始终不出现的解决方法,涉及Android界面布局相关属性与状态设置操作技巧,需要的朋友可以参考下
    2019-03-03
  • Android 创建与解析XML(五)——详解Dom4j方式

    Android 创建与解析XML(五)——详解Dom4j方式

    本篇文章主要介绍了Android创建与解析XML(二)——详解Dom4j方式,这里整理了详细的代码,有需要的小伙伴可以参考下。
    2016-11-11
  • Android自定义谷歌风格ProgressBar

    Android自定义谷歌风格ProgressBar

    这篇文章主要为大家详细介绍了Android自定义谷歌风格ProgressBar的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-02-02
  • Android 实现仿支付宝的密码均分输入框

    Android 实现仿支付宝的密码均分输入框

    这篇文章主要介绍了Android 实现仿支付宝的密码均分输入框的相关资料,需要的朋友可以参考下
    2017-06-06
  • Android 动画之TranslateAnimation应用详解

    Android 动画之TranslateAnimation应用详解

    本节讲解TranslateAnimation动画,TranslateAnimation比较常用,比如QQ,网易新闻菜单条的动画,就可以用TranslateAnimation实现,本文将详细介绍通过TranslateAnimation 来定义动画,需要的朋友可以参考下
    2012-12-12
  • Android开发之WebView组件的使用解析

    Android开发之WebView组件的使用解析

    WebView 类是 WebKit 模块 Java 层的视图类, 所有需要使用 Web 浏览功能的Android应用程序都要创建该视图对象显示和处理请求的网络资源,接下来将详细介绍,需要了解的朋友可以参考下
    2012-12-12
  • android USB如何修改VID具体实现

    android USB如何修改VID具体实现

    在android 设备的Linux 内核中把 USB 驱动的 PID VID 修改以后,也许之前的adb工具就不能识别设备了,会打印出"device not found"的提示
    2013-06-06
  • 简单实现Android端搜索框示例详解

    简单实现Android端搜索框示例详解

    这篇文章主要为大家介绍了简单实现Android端搜索框示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-11-11

最新评论