使用 html5 在网页录音和保存

2018年04月25日 15:09 · 阅读(2174) ·

[目录]

本文引用自

这篇的内容都不是我写的,我只是把例子自己用 MVC3 做了一遍,主要是下面这位大神实现的

HTML5网页录音和压缩,边猜边做..(附源码)

源码 Demo

见附件

HZRecorder.js

  1. (function (window) {
  2. //兼容
  3. window.URL = window.URL || window.webkitURL;
  4. navigator.getUserMedia = navigator.getUserMedia || navigator.webkitGetUserMedia || navigator.mozGetUserMedia || navigator.msGetUserMedia;
  5. var HZRecorder = function (stream, config) {
  6. config = config || {};
  7. config.sampleBits = config.sampleBits || 8; //采样数位 8, 16
  8. config.sampleRate = config.sampleRate || (44100 / 6); //采样率(1/6 44100)
  9. var context = new (window.webkitAudioContext || window.AudioContext)();
  10. var audioInput = context.createMediaStreamSource(stream);
  11. var createScript = context.createScriptProcessor || context.createJavaScriptNode;
  12. var recorder = createScript.apply(context, [4096, 1, 1]);
  13. var audioData = {
  14. size: 0 //录音文件长度
  15. , buffer: [] //录音缓存
  16. , inputSampleRate: context.sampleRate //输入采样率
  17. , inputSampleBits: 16 //输入采样数位 8, 16
  18. , outputSampleRate: config.sampleRate //输出采样率
  19. , oututSampleBits: config.sampleBits //输出采样数位 8, 16
  20. , input: function (data) {
  21. this.buffer.push(new Float32Array(data));
  22. this.size += data.length;
  23. }
  24. , compress: function () { //合并压缩
  25. //合并
  26. var data = new Float32Array(this.size);
  27. var offset = 0;
  28. for (var i = 0; i < this.buffer.length; i++) {
  29. data.set(this.buffer[i], offset);
  30. offset += this.buffer[i].length;
  31. }
  32. //压缩
  33. var compression = parseInt(this.inputSampleRate / this.outputSampleRate);
  34. var length = data.length / compression;
  35. var result = new Float32Array(length);
  36. var index = 0, j = 0;
  37. while (index < length) {
  38. result[index] = data[j];
  39. j += compression;
  40. index++;
  41. }
  42. return result;
  43. }
  44. , encodeWAV: function () {
  45. var sampleRate = Math.min(this.inputSampleRate, this.outputSampleRate);
  46. var sampleBits = Math.min(this.inputSampleBits, this.oututSampleBits);
  47. var bytes = this.compress();
  48. var dataLength = bytes.length * (sampleBits / 8);
  49. var buffer = new ArrayBuffer(44 + dataLength);
  50. var data = new DataView(buffer);
  51. var channelCount = 1;//单声道
  52. var offset = 0;
  53. var writeString = function (str) {
  54. for (var i = 0; i < str.length; i++) {
  55. data.setUint8(offset + i, str.charCodeAt(i));
  56. }
  57. }
  58. // 资源交换文件标识符
  59. writeString('RIFF'); offset += 4;
  60. // 下个地址开始到文件尾总字节数,即文件大小-8
  61. data.setUint32(offset, 36 + dataLength, true); offset += 4;
  62. // WAV文件标志
  63. writeString('WAVE'); offset += 4;
  64. // 波形格式标志
  65. writeString('fmt '); offset += 4;
  66. // 过滤字节,一般为 0x10 = 16
  67. data.setUint32(offset, 16, true); offset += 4;
  68. // 格式类别 (PCM形式采样数据)
  69. data.setUint16(offset, 1, true); offset += 2;
  70. // 通道数
  71. data.setUint16(offset, channelCount, true); offset += 2;
  72. // 采样率,每秒样本数,表示每个通道的播放速度
  73. data.setUint32(offset, sampleRate, true); offset += 4;
  74. // 波形数据传输率 (每秒平均字节数) 单声道×每秒数据位数×每样本数据位/8
  75. data.setUint32(offset, channelCount * sampleRate * (sampleBits / 8), true); offset += 4;
  76. // 快数据调整数 采样一次占用字节数 单声道×每样本的数据位数/8
  77. data.setUint16(offset, channelCount * (sampleBits / 8), true); offset += 2;
  78. // 每样本数据位数
  79. data.setUint16(offset, sampleBits, true); offset += 2;
  80. // 数据标识符
  81. writeString('data'); offset += 4;
  82. // 采样数据总数,即数据总大小-44
  83. data.setUint32(offset, dataLength, true); offset += 4;
  84. // 写入采样数据
  85. if (sampleBits === 8) {
  86. for (var i = 0; i < bytes.length; i++, offset++) {
  87. var s = Math.max(-1, Math.min(1, bytes[i]));
  88. var val = s < 0 ? s * 0x8000 : s * 0x7FFF;
  89. val = parseInt(255 / (65535 / (val + 32768)));
  90. data.setInt8(offset, val, true);
  91. }
  92. } else {
  93. for (var i = 0; i < bytes.length; i++, offset += 2) {
  94. var s = Math.max(-1, Math.min(1, bytes[i]));
  95. data.setInt16(offset, s < 0 ? s * 0x8000 : s * 0x7FFF, true);
  96. }
  97. }
  98. return new Blob([data], { type: 'audio/wav' });
  99. }
  100. };
  101. //开始录音
  102. this.start = function () {
  103. audioInput.connect(recorder);
  104. recorder.connect(context.destination);
  105. }
  106. //停止
  107. this.stop = function () {
  108. recorder.disconnect();
  109. }
  110. //获取音频文件
  111. this.getBlob = function () {
  112. this.stop();
  113. return audioData.encodeWAV();
  114. }
  115. //回放
  116. this.play = function (audio) {
  117. audio.src = window.URL.createObjectURL(this.getBlob());
  118. }
  119. //上传
  120. this.upload = function (url, callback) {
  121. var fd = new FormData();
  122. fd.append("audioData", this.getBlob());
  123. var xhr = new XMLHttpRequest();
  124. if (callback) {
  125. xhr.upload.addEventListener("progress", function (e) {
  126. callback('uploading', e);
  127. }, false);
  128. xhr.addEventListener("load", function (e) {
  129. callback('ok', e);
  130. }, false);
  131. xhr.addEventListener("error", function (e) {
  132. callback('error', e);
  133. }, false);
  134. xhr.addEventListener("abort", function (e) {
  135. callback('cancel', e);
  136. }, false);
  137. }
  138. xhr.open("POST", url);
  139. xhr.send(fd);
  140. }
  141. //音频采集
  142. recorder.onaudioprocess = function (e) {
  143. audioData.input(e.inputBuffer.getChannelData(0));
  144. //record(e.inputBuffer.getChannelData(0));
  145. }
  146. };
  147. //抛出异常
  148. HZRecorder.throwError = function (message) {
  149. alert(message);
  150. throw new function () { this.toString = function () { return message; } }
  151. }
  152. //是否支持录音
  153. HZRecorder.canRecording = (navigator.getUserMedia != null);
  154. //获取录音机
  155. HZRecorder.get = function (callback, config) {
  156. if (callback) {
  157. if (navigator.getUserMedia) {
  158. navigator.getUserMedia(
  159. { audio: true } //只启用音频
  160. , function (stream) {
  161. var rec = new HZRecorder(stream, config);
  162. callback(rec);
  163. }
  164. , function (error) {
  165. switch (error.code || error.name) {
  166. case 'PERMISSION_DENIED':
  167. case 'PermissionDeniedError':
  168. HZRecorder.throwError('用户拒绝提供信息。');
  169. break;
  170. case 'NOT_SUPPORTED_ERROR':
  171. case 'NotSupportedError':
  172. HZRecorder.throwError('浏览器不支持硬件设备。');
  173. break;
  174. case 'MANDATORY_UNSATISFIED_ERROR':
  175. case 'MandatoryUnsatisfiedError':
  176. HZRecorder.throwError('无法发现指定的硬件设备。');
  177. break;
  178. default:
  179. HZRecorder.throwError('无法打开麦克风。异常信息:' + (error.code || error.name));
  180. break;
  181. }
  182. });
  183. } else {
  184. HZRecorder.throwErr('当前浏览器不支持录音功能。'); return;
  185. }
  186. }
  187. }
  188. window.HZRecorder = HZRecorder;
  189. })(window);

HTML

  1. @{
  2. Layout = null;
  3. }
  4. <!DOCTYPE html>
  5. <html xmlns="http://www.w3.org/1999/xhtml">
  6. <head>
  7. <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
  8. <title>HZRecorder</title>
  9. </head>
  10. <body>
  11. <div>
  12. <audio controls autoplay></audio>
  13. <input onclick="startRecording()" type="button" value="录音" />
  14. <input onclick="stopRecording()" type="button" value="停止" />
  15. <input onclick="playRecording()" type="button" value="播放" />
  16. <input onclick="uploadAudio()" type="button" value="提交" />
  17. </div>
  18. <script type="text/javascript" src="@Url.Content("~/HZRecorder/HZRecorder.js")"></script>
  19. <script>
  20. var recorder;
  21. var audio = document.querySelector('audio');
  22. function startRecording() {
  23. HZRecorder.get(function (rec) {
  24. recorder = rec;
  25. recorder.start();
  26. });
  27. }
  28. function stopRecording() {
  29. recorder.stop();
  30. }
  31. function playRecording() {
  32. recorder.play(audio);
  33. }
  34. function uploadAudio() {
  35. recorder.upload("@Url.Action("SaveFile","Default")", function (state, e) {
  36. switch (state) {
  37. case 'uploading':
  38. //var percentComplete = Math.round(e.loaded * 100 / e.total) + '%';
  39. break;
  40. case 'ok':
  41. debugger
  42. //alert(e.target.responseText);
  43. alert("上传成功");
  44. alert(e.target.responseText.AudioSrc);
  45. break;
  46. case 'error':
  47. alert("上传失败");
  48. break;
  49. case 'cancel':
  50. alert("上传被取消");
  51. break;
  52. }
  53. });
  54. }
  55. </script>
  56. </body>
  57. </html>

Controller

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Web;
  5. using System.Web.Mvc;
  6. namespace RecordDemo.Controllers
  7. {
  8. public class DefaultController : Controller
  9. {
  10. public ActionResult Index()
  11. {
  12. return View();
  13. }
  14. public ActionResult HZRecorder()
  15. {
  16. return View();
  17. }
  18. [HttpPost]
  19. public JsonResult SaveFile(HttpPostedFileBase f)
  20. {
  21. var files = Request.Files;
  22. if (files.Count == 0)
  23. return Json(new { Success = false, Message = "提示:没有要上传的文件!" }, JsonRequestBehavior.AllowGet);
  24. string savePath = "/Content/Audio/";
  25. string audioUrl = "http://" + Request.Url.Authority + savePath;
  26. foreach (string file in files)
  27. {
  28. //获取文件信息
  29. var curFile = Request.Files[file];
  30. string fileName = "1.wav";
  31. //保存文件到服务器磁盘
  32. curFile.SaveAs(Server.MapPath("~" + savePath) + fileName);
  33. audioUrl += fileName;
  34. }
  35. return Json(new { Success = true, AudioSrc = audioUrl }, JsonRequestBehavior.AllowGet);
  36. }
  37. }
  38. }

相关附件 RecordDemo-2018.4.25 | 大小:0.03M | 下载