需要全部源码请点赞关注收藏后评论区留下QQ~~~

一、需求分析

用户不仅能在平台上收听音频,还能成为内容创作者,总之长音频分享平台需要满足两种角色的使用:一种是作为内容创作者发布自己的音频,另一种是作为用户欣赏平台上的已有音频

二、功能分析

长音频分享主要集成了如下App技术

1:网格控件

长音频分享首页的栏目列表,以网格形式排列

2:属性动画

在音频录制过程中 上方的风车图标持续旋转

3:弹幕动画

在音频收听界面,可以划过弹幕

4:音频控制条

无论是用户收听音频还是创作者试听音频 都需要音频控制条协助播音

5:JNI接口

创作者录制的原始音频要求转成MP3格式 需要借助第三方的LAME库

6:网络通信框架

上传音频信息与获取音频列表均与后端交互

7:图片加载框架

音频封面来自Web服务  建议利用Glide框架加载网络图片

8:Socket通信

采取Socket通信与后端服务器交互

9:移动数据格式JSON

传输评论信息时  需要把消息结构封装为JSON结构

下面介绍一下源码中各模块之间的关系

StoryViewActivity  长音频分享的首页列表

StoryListenActivity  说书音频的欣赏页面

StoryTakeActivity  音频录制页面

StoryEditActivity  音频信息的编辑页面

AudioLoadTask 首页音频列表的加载任务

与此同时 长音频分享还需要与之配合的HTTP服务器和Socket服务器 此处不再赘述

三、演示效果

演示视频已上传至个人主页 有需要可以前往观看

首页如下

点击下方的话筒图标 跳到音频的初始界面 可以开始录音

同时可以给自己录制的音频添加图案和音频描述

同时能在多台手机上收听和发出弹幕

四、代码

部分源码如下 需要全部代码以及依赖请点赞关注收藏后评论区留下QQ~~~

编辑类

package com.example.audio;import androidx.appcompat.app.AppCompatActivity;import android.app.ProgressDialog;
import android.content.Intent;
import android.graphics.Bitmap;
import android.net.Uri;
import android.os.Bundle;
import android.os.Environment;
import android.text.TextUtils;
import android.widget.EditText;
import android.widget.ImageView;
import android.widget.TextView;
import android.widget.Toast;import com.example.audio.bean.CommitResponse;
import com.example.audio.constant.UrlConstant;
import com.example.audio.util.BitmapUtil;
import com.example.audio.util.DateUtil;
import com.example.audio.widget.AudioController;
import com.google.gson.Gson;import java.io.File;
import java.io.IOException;import okhttp3.Call;
import okhttp3.Callback;
import okhttp3.MediaType;
import okhttp3.MultipartBody;
import okhttp3.OkHttpClient;
import okhttp3.Request;
import okhttp3.RequestBody;
import okhttp3.Response;public class StoryEditActivity extends AppCompatActivity {private final static String TAG = "StoryEditActivity";private ImageView iv_cover; // 声明一个图像视图对象private EditText et_artist; // 声明一个编辑框对象private EditText et_title; // 声明一个编辑框对象private EditText et_desc; // 声明一个编辑框对象private AudioController ac_play; // 声明一个音频控制条对象private String mAudioPath; // 音频文件路径private int CHOOSE_CODE = 3; // 只在相册挑选图片的请求码private Bitmap mCoverBitmap; // 声明一个位图对象private ProgressDialog mDialog; // 声明一个对话框对象@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_story_edit);findViewById(R.id.iv_back).setOnClickListener(v -> finish());TextView tv_title = findViewById(R.id.tv_title);tv_title.setText("编辑说书音频");iv_cover = findViewById(R.id.iv_cover);et_artist = findViewById(R.id.et_artist);et_title = findViewById(R.id.et_title);et_desc = findViewById(R.id.et_desc);ac_play = findViewById(R.id.ac_play);findViewById(R.id.iv_cover).setOnClickListener(v -> {// 创建一个内容获取动作的意图(准备跳到系统相册)Intent albumIntent = new Intent(Intent.ACTION_GET_CONTENT);albumIntent.putExtra(Intent.EXTRA_ALLOW_MULTIPLE, false); // 是否允许多选albumIntent.setType("image/*"); // 类型为图像startActivityForResult(albumIntent, CHOOSE_CODE); // 打开系统相册});findViewById(R.id.btn_upload).setOnClickListener(v -> uploadAudio());mAudioPath = getIntent().getStringExtra("audio_path");ac_play.prepare(mAudioPath); // 准备播放指定路径的音频}@Overrideprotected void onActivityResult(int requestCode, int resultCode, Intent intent) {super.onActivityResult(requestCode, resultCode, intent);if (resultCode == RESULT_OK && requestCode == CHOOSE_CODE) { // 从相册返回if (intent.getData() != null) { // 从相册选择一张照片Uri uri = intent.getData(); // 获得已选择照片的路径对象// 根据指定图片的uri,获得自动缩小后的位图对象mCoverBitmap = BitmapUtil.getAutoZoomImage(this, uri);iv_cover.setImageBitmap(mCoverBitmap); // 设置图像视图的位图对象}}}// 执行音频上传动作private void uploadAudio() {String artist = et_artist.getText().toString();String title = et_title.getText().toString();String desc = et_desc.getText().toString();if (TextUtils.isEmpty(artist)) {Toast.makeText(this, "请先输入音频作者名称", Toast.LENGTH_SHORT).show();return;}if (TextUtils.isEmpty(title)) {Toast.makeText(this, "请先输入音频的标题", Toast.LENGTH_SHORT).show();return;}if (TextUtils.isEmpty(desc)) {Toast.makeText(this, "请先输入音频的描述", Toast.LENGTH_SHORT).show();return;}// 弹出进度对话框mDialog = ProgressDialog.show(this, "请稍候", "正在上传音频信息......");String coverPath = String.format("%s/%s.jpg",getExternalFilesDir(Environment.DIRECTORY_DOWNLOADS).toString(),DateUtil.getNowDateTime());BitmapUtil.saveImage(coverPath, mCoverBitmap); // 把位图保存为图片文件// 下面把音频信息(包含封面)提交给HTTP服务端MultipartBody.Builder builder = new MultipartBody.Builder();// 往建造器对象添加文本格式的分段数据builder.addFormDataPart("artist", artist); // 作者builder.addFormDataPart("title", title); // 标题builder.addFormDataPart("desc", desc); // 描述// 往建造器对象添加图像格式的分段数据builder.addFormDataPart("cover", coverPath.substring(coverPath.lastIndexOf("/")),RequestBody.create(new File(coverPath), MediaType.parse("image/*")));// 往建造器对象添加音频格式的分段数据builder.addFormDataPart("audio", mAudioPath.substring(mAudioPath.lastIndexOf("/")),RequestBody.create(new File(mAudioPath), MediaType.parse("audio/*")));RequestBody body = builder.build(); // 根据建造器生成请求结构OkHttpClient client = new OkHttpClient(); // 创建一个okhttp客户端对象// 创建一个POST方式的请求结构Request request = new Request.Builder().post(body).url(UrlConstant.HTTP_PREFIX+"commitAudio").build();Call call = client.newCall(request); // 根据请求结构创建调用对象// 加入HTTP请求队列。异步调用,并设置接口应答的回调方法call.enqueue(new Callback() {@Overridepublic void onFailure(Call call, IOException e) { // 请求失败// 回到主线程操纵界面runOnUiThread(() ->  {mDialog.dismiss(); // 关闭进度对话框Toast.makeText(StoryEditActivity.this,"上传音频信息出错:"+e.getMessage(), Toast.LENGTH_SHORT).show();});}@Overridepublic void onResponse(Call call, final Response response) throws IOException { // 请求成功String resp = response.body().string();CommitResponse commitResponse = new Gson().fromJson(resp, CommitResponse.class);// 回到主线程操纵界面runOnUiThread(() -> {mDialog.dismiss(); // 关闭进度对话框if ("0".equals(commitResponse.getCode())) {finishUpload(); // 结束音频上传动作} else {Toast.makeText(StoryEditActivity.this, "上传音频信息失败:"+commitResponse.getDesc(), Toast.LENGTH_SHORT).show();}});}});}// 结束音频上传动作private void finishUpload() {Toast.makeText(this, "成功上传您的说书音频", Toast.LENGTH_SHORT).show();// 下面重新打开音频列表浏览界面Intent intent = new Intent(this, StoryViewActivity.class);// 设置启动标志:跳转到新页面时,栈中的原有实例都被清空,同时开辟新任务的活动栈intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TASK | Intent.FLAG_ACTIVITY_NEW_TASK);startActivity(intent);}@Overrideprotected void onResume() {super.onResume();ac_play.resume(); // 恢复播放}@Overrideprotected void onPause() {super.onPause();ac_play.pause(); // 暂停播放}@Overrideprotected void onDestroy() {super.onDestroy();ac_play.release(); // 释放播放资源}}

收听类

package com.example.audio;import androidx.appcompat.app.AppCompatActivity;import android.os.Bundle;
import android.text.TextUtils;
import android.util.Log;
import android.view.WindowManager;
import android.widget.EditText;
import android.widget.ImageView;
import android.widget.TextView;
import android.widget.Toast;import com.bumptech.glide.Glide;
import com.bumptech.glide.load.resource.bitmap.RoundedCorners;
import com.bumptech.glide.request.RequestOptions;
import com.example.audio.bean.AudioInfo;
import com.example.audio.bean.JoinInfo;
import com.example.audio.bean.MessageInfo;
import com.example.audio.constant.UrlConstant;
import com.example.audio.util.DateUtil;
import com.example.audio.util.SocketUtil;
import com.example.audio.util.Utils;
import com.example.audio.util.ViewUtil;
import com.example.audio.widget.AudioController;
import com.example.audio.widget.BarrageView;
import com.google.gson.Gson;import org.json.JSONObject;import io.socket.client.Socket;public class StoryListenActivity extends AppCompatActivity {private final static String TAG = "StoryListenActivity";private TextView tv_title; // 声明一个文本视图对象private TextView tv_artist; // 声明一个文本视图对象private ImageView iv_cover; // 声明一个图像视图对象private TextView tv_desc; // 声明一个文本视图对象private BarrageView bv_comment; // 声明一个弹幕视图对象private AudioController ac_play; // 声明一个音频控制条对象private EditText et_input; // 声明一个编辑框对象private String mSelfName, mGroupName; // 自己名称,群组名称private Socket mSocket; // 声明一个套接字对象@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_story_listen);getWindow().addFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON); // 保持屏幕常亮initView(); // 初始化视图playStory(); // 播放故事音频initSocket(); // 初始化套接字}// 初始化视图private void initView() {findViewById(R.id.iv_back).setOnClickListener(v -> finish());tv_title = findViewById(R.id.tv_title);tv_artist = findViewById(R.id.tv_artist);iv_cover = findViewById(R.id.iv_cover);tv_desc = findViewById(R.id.tv_desc);bv_comment = findViewById(R.id.bv_comment);ac_play = findViewById(R.id.ac_play);et_input = findViewById(R.id.et_input);findViewById(R.id.btn_send).setOnClickListener(v -> sendMessage());}// 播放故事音频private void playStory() {AudioInfo audio = (AudioInfo) getIntent().getSerializableExtra("audio_info");mSelfName = DateUtil.getFullDateTime();mGroupName = audio.getTitle();tv_artist.setText(String.format("《%s》%s", audio.getTitle(), audio.getArtist()));tv_title.setText(audio.getTitle());tv_desc.setText(audio.getDesc());// 使用Glide加载圆角矩形裁剪后的故事封面RoundedCorners roundedCorners = new RoundedCorners(Utils.dip2px(this, 30));RequestOptions options = RequestOptions.bitmapTransform(roundedCorners);Glide.with(this).load(UrlConstant.HTTP_PREFIX+audio.getCover()).apply(options).into(iv_cover);ac_play.prepareAsync(UrlConstant.HTTP_PREFIX+audio.getAudio()); // 准备播放指定链接的网络音频}// 初始化套接字private void initSocket() {mSocket = MainApplication.getInstance().getSocket();mSocket.connect(); // 建立Socket连接// 等待接收弹幕消息mSocket.on("receive_group_message", (args) -> {JSONObject json = (JSONObject) args[0];MessageInfo message = new Gson().fromJson(json.toString(), MessageInfo.class);// 往故事窗口添加弹幕评论runOnUiThread(() -> bv_comment.addComment(message.content));});// 下面向Socket服务器发送入群通知JoinInfo joinInfo = new JoinInfo(mSelfName, mGroupName);SocketUtil.emit(mSocket, "join_group", joinInfo);}// 发送评论消息private void sendMessage() {String content = et_input.getText().toString();if (TextUtils.isEmpty(content)) {Toast.makeText(this, "请输入评论消息", Toast.LENGTH_SHORT).show();return;}et_input.setText("");ViewUtil.hideOneInputMethod(this, et_input); // 隐藏软键盘bv_comment.addComment(content); // 给弹幕视图添加评论// 下面向Socket服务器发送群消息MessageInfo message = new MessageInfo(mSelfName, mGroupName, content);SocketUtil.emit(mSocket, "send_group_message", message);}@Overrideprotected void onResume() {super.onResume();ac_play.resume(); // 恢复播放}@Overrideprotected void onPause() {super.onPause();ac_play.pause(); // 暂停播放}@Overrideprotected void onDestroy() {super.onDestroy();ac_play.release(); // 释放播放资源if (mSocket.connected()) { // 已经连上Socket服务器// 下面向Socket服务器发送退群通知JoinInfo joinInfo = new JoinInfo(mSelfName, mGroupName);SocketUtil.emit(mSocket, "leave_group", joinInfo);mSocket.off("receive_group_message"); // 取消接收弹幕消息mSocket.disconnect(); // 断开Socket连接}}}

创作不易 觉得有帮助请点赞关注收藏~~

Android App开发实战项目之仿喜马拉雅的听说书App实现(超详细 附源码和演示视频)相关推荐

  1. 【Android App】实战项目之实现你问我答的智能语音机器人(超详细 附源码和演示视频)

    需要全部代码请点赞关注收藏后评论区留言私信~~~ 一.需求描述 想必大家都见过商场里的智能语音机器人,你对它提问时它可以自动回答你的问题,接下来我们也实现这样一个机器人,它依靠语音技术完成问询服务 基 ...

  2. 【Android +Tensroflow Lite】实现从基于机器学习语音中识别指令讲解及实战(超详细 附源码和演示视频)

    需要源码和配置文件请点赞关注收藏后评论区留言~~~ 一.基于机器学习的语音推断 Tensorflow基于分层和模块化的设计思想,整个框架以C语言的编程接口为界,分为前端和后端两大部分 Tensorfl ...

  3. 【Android App】实战项目之使用OpenCV人脸识别实现找人功能(附源码和演示 超详细)

    需要全部代码请点赞关注收藏后评论区留言私信~~~ 人脸识别自古有之,每当官府要捉拿某人时,便在城墙贴出通缉告示并附上那人的肖像.只是该办法依赖人们的回忆与主观判断,指认结果多有出入,算不上什么先进. ...

  4. Android 开发人脸识别之自动识别验证码功能讲解及实现(超详细 附源码)

    需要源码和图片集请点赞关注收藏后评论区留下QQ或者私信~~~ 一.自动识别验证码 验证码图片中最简单的是数字验证码,一张再普通不过的验证码拿到之后要进行以下步骤的处理 1:首先对图片适当裁剪,先去掉外 ...

  5. Android App开发实战项目之仿手机QQ动感影集动画播放(附源码和演示视频 可直接使用)

    需要图片集和源码请点赞关注收藏后评论区留言~~~ 动感影集就是只要用户添加一张图片,动感影集就能给每张图片渲染不同的动画效果,让原本静止的图片变得活泼起来,辅以各种精致的动画特效,营造一种赏心悦目的感 ...

  6. 【Android App】实战项目之仿微信的私信和群聊App(附源码和演示视频 超详细必看)

    需要全部代码请点赞关注收藏后评论区留言私信~~~ 手机最开始用于通话,后来增加了短信功能,初步满足了人与人之间的沟通需求.然而短信只能发文字,于是出现了能够发图片的彩信,但不管短信还是彩信,资费都太贵 ...

  7. Android App开发实战项目之模仿美图秀秀的抠图工具(附源码和演示视频 简单易懂 可直接使用)

    需要图片集和源码请点赞关注收藏后评论区留言~~~ 所谓抠图神器,就是从一副图片中扣出用户想要的某块区域 一.需求描述 美图的修图功能如此强大,离不开专业的图片加工技术,抠图便是其中重要的一项功能.在A ...

  8. Android App开发语音处理之系统自带的语音引擎、文字转语音、语音识别的讲解及实战(超详细 附源码)

    需要源码请点赞关注收藏后评论区留下QQ~~~ 一.系统自带的语音引擎 语音播报的本质是将书面文字转换成自然语言的音频流,这个转换操作被称作语音合成,又称TTS(从文本到语音)在转换过程中,为了避免机械 ...

  9. 【Android App】实战项目之虚拟现实(VR)的全景相册(附源码和演示视频 可用于学习和大作业)

    需要源码请点赞关注收藏后评论区留言私信~~~ 不管是绘画还是摄影,都是把三维的物体投影到平面上,其实仍旧呈现二维的模拟画面. 随着科技的发展,传统的成像手段越来越凸显出局限性,缘由在于人们需要一种更逼 ...

最新文章

  1. 【SSH网上商城项目实战23】完成在线支付功能
  2. JS URL 编码 PHP 解码{%u5F00%u53D1}
  3. 数组、ArrayList、链表、LinkedList
  4. 深入理解 JVM Class文件格式(一)
  5. mysql客户端安装错误_windows下mysql 5.7以上版本安装及遇到的问题
  6. 使用X.509数字证书加密解密实务(一)-- 证书的获得和管理
  7. hdu5468 Puzzled Elena
  8. 2003服务器远程桌面连不上解决办法
  9. 【错误纠正】关于文章《绕开数学,讲讲信息论》
  10. eclipse启动提示java,Eclipse启动时报错-JSP教程,Java技巧及代码
  11. python常用进制转换の方法
  12. 极致业务基础开发平台
  13. mysql教程丿it教程网_1.0MySQL基础入门【DQL部分】
  14. 深度学习:bert embedding用法详解
  15. [BZOJ4621]Tc605
  16. 路由器与交换机怎么插线_网络设备:中继器、集线器、网桥、交换机、路由器、网关的超全总结!...
  17. S71200PLC程序博图V14 西门子博图编写
  18. html5 手机uc浏览器 复制链接,UC浏览器里任意复制粘贴文本的方法
  19. 愿所有我和码农们 printf(“前程似锦”)
  20. 64马8赛道取前4问题

热门文章

  1. 将XML格式化后输出为String
  2. java三角形剪角_完成撤销交易后,作废的银行汇票四联剪角,加盖“作废”戳记。...
  3. python从菜鸟到高手李宁pdf_尹成学院-Python从菜鸟到高手编程实战【已完结 28G】...
  4. 卷积神经网络图像尺寸预处理-----图像裁剪
  5. Linux压缩pdf
  6. (19)网络安全:WAF你绕过去了嘛?没有撤退可言。
  7. 全球与中国草坪刷市场现状及未来发展趋势2022-2028
  8. 基于PHP+小程序(MINA框架)+Mysql数据库的驾校考试小程序系统设计与实现
  9. 程序员因为女孩而美丽!
  10. JSPatch 尝试