├── .gitattributes
├── .gitignore
├── README.md
├── config.rb
├── index.html
├── lib
├── wx-audio.css
└── wx-audio.js
├── package-lock.json
├── package.json
├── src
├── assets
│ ├── pause.png
│ └── playing.gif
├── index.js
├── lib
│ ├── wx-audio.js
│ └── wx-audio.less
└── style.less
├── tea.yaml
├── webpack.config.js
└── yarn.lock
/.gitattributes:
--------------------------------------------------------------------------------
1 | *.js linguist-language=javascript
2 | *.css linguist-language=javascript
3 | *.html linguist-language=javascript
4 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | .DS_Store
2 | node_modules/
3 | dist/
4 | npm-debug.log*
5 | yarn-debug.log*
6 | yarn-error.log*
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # wx-audio
2 | 微信公众号音乐播放器
3 |
4 | #### 基于原生js写的一款仿微信公众号的音乐组件
5 |
6 | ## 演示地址
7 | http://www.daiwei.org/components/wx-audio/
8 |
9 | ## 手机预览
10 |
11 | ### 安装
12 | npm 安装
13 | ```js
14 | npm install @dw/wx-audio
15 | ```
16 |
17 | ### 引入
18 | ```js
19 | import WxAudio from '@dw/wx-audio/lib/wx-audio.css'
20 | import WxAudio from '@dw/wx-audio'
21 | ```
22 |
23 | 普通资源引入,先将lib文件下的目录放到自己的项目中,然后根据路径引入
24 | ```html
25 |
26 |
27 | ```
28 |
29 | ### 实例化 音乐组件
30 | ```js
31 | var wxAudio = new WxAudio({
32 | ele: '#textaudio1',
33 | title: '河山大好',
34 | disc: '许嵩',
35 | src: 'http://oiq8j9er1.bkt.clouddn.com/%E8%AE%B8%E5%B5%A9%20-%20%E6%B2%B3%E5%B1%B1%E5%A4%A7%E5%A5%BD1.mp3',
36 | width: '320px',
37 | autoplay: true,
38 | loop: true,
39 | ended: function () {
40 | alert('播放结束')
41 | }
42 | });
43 | ```
44 |
45 | ### 属性
46 | - ele dom对象
47 | - title 音乐标题
48 | - disc 描述,可以当做显示歌手名称
49 | - src 地址
50 | - width 显示的宽度
51 | - loop 是否循环(当循环状态下时候,ended事件不会执行)
52 | - ended 方法,是一个回调,可在音乐播放结束之后执行其他的代码
53 | - autoplay 属性, 是否自动播放,这个只能在微信浏览器以及只设置静态src才能自动播放
54 | ### 方法
55 | ```js
56 | // 实例化的wxAudio可以这样操作
57 | wxAudio.audioPlay() // 播放
58 |
59 | wxAudio.audioPause() // 暂停
60 |
61 | wxAudio.audioPlayPause() // 播放暂停
62 |
63 | wxAudio.showLoading(bool) //显示加载状态 参数bool
64 |
65 | wxAudio.audioCut(src,title,disc) 实现音频切换的效果
66 | ```
67 |
68 | ### 新增 音乐组件切歌方法
69 | 通过实例化的wxAudio的audioCut(src,title,disc) 实现音频切换的效果 示例代码如下
70 | ```js
71 | var src = 'http://oiq8j9er1.bkt.clouddn.com/%E6%9E%97%E4%BF%8A%E6%9D%B0%20-%20%E5%A5%B9%E8%AF%B41.mp3'
72 | var title = '她说'
73 | var disc = '林俊杰'
74 | wxAudio.audioCut(src, title, disc)
75 | ```
76 |
--------------------------------------------------------------------------------
/config.rb:
--------------------------------------------------------------------------------
1 | require 'compass/import-once/activate'
2 | # Require any additional compass plugins here.
3 |
4 | # Set this to the root of your project when deployed:
5 | http_path = "/"
6 | css_dir = "css"
7 | sass_dir = "sass"
8 | images_dir = "images"
9 | javascripts_dir = "js"
10 |
11 | # You can select your preferred output style here (can be overridden via the command line):
12 | # output_style = :expanded or :nested or :compact or :compressed
13 |
14 | # To enable relative paths to assets via compass helper functions. Uncomment:
15 | # relative_assets = true
16 |
17 | # To disable debugging comments that display the original location of your selectors. Uncomment:
18 | # line_comments = false
19 |
20 |
21 | # If you prefer the indented syntax, you might want to regenerate this
22 | # project again passing --syntax sass, or you can uncomment this:
23 | # preferred_syntax = :sass
24 | # and then run:
25 | # sass-convert -R --from scss --to sass sass scss && rm -rf sass && mv scss sass
26 |
--------------------------------------------------------------------------------
/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | Wx-audio
6 |
8 |
9 |
10 | es6 wx-audio
11 |
12 | 播放
13 | 暂停
14 | 切歌
15 |
16 |
--------------------------------------------------------------------------------
/lib/wx-audio.css:
--------------------------------------------------------------------------------
1 | @keyframes slidein {
2 | 0% {
3 | -webkit-transform: translateX(-50%);
4 | -moz-transform: translateX(-50%);
5 | -ms-transform: translateX(-50%);
6 | -o-transform: translateX(-50%);
7 | transform: translateX(-50%);
8 | }
9 | 100% {
10 | -webkit-transform: translateX(0);
11 | -moz-transform: translateX(0);
12 | -ms-transform: translateX(0);
13 | -o-transform: translateX(0);
14 | transform: translateX(0);
15 | }
16 | }
17 | .wx-audio-content {
18 | user-select: none;
19 | width: 100%;
20 | display: flex;
21 | align-items: center;
22 | justify-content: center;
23 | min-height: auto;
24 | padding: 8px 12px;
25 | border: 1px solid #efefef;
26 | margin: 0 auto;
27 | font-size: 0;
28 | box-sizing: border-box;
29 | font-family: 'PingFangSC-Regular';
30 | -webkit-tap-highlight-color: transparent;
31 | background: #fdfdfd;
32 | }
33 | .wx-audio-content p {
34 | margin: 0;
35 | }
36 | .wx-audio-content .wx-audio-info {
37 | display: none;
38 | }
39 | .wx-audio-content .wx-audio-left {
40 | display: inline-block;
41 | width: 42px;
42 | height: 100%;
43 | }
44 | .wx-audio-content .wx-audio-left .wx-audio-state {
45 | width: 100%;
46 | height: 100%;
47 | }
48 | .wx-audio-content .wx-audio-right {
49 | display: inline-block;
50 | width: calc(100% - 42px);
51 | font-size: 16px;
52 | box-sizing: border-box;
53 | padding-left: 10px;
54 | }
55 | .wx-audio-content .wx-audio-right .wx-audio-title {
56 | padding-bottom: 6px;
57 | width: 100%;
58 | font-size: 16px;
59 | font-weight: 400;
60 | text-overflow: ellipsis;
61 | overflow: hidden;
62 | white-space: nowrap;
63 | }
64 | .wx-audio-content .wx-audio-right .wx-audio-disc {
65 | width: 100%;
66 | text-overflow: ellipsis;
67 | overflow: hidden;
68 | white-space: nowrap;
69 | padding-bottom: 10px;
70 | font-size: 12px;
71 | color: #8c8c8c;
72 | }
73 | .wx-audio-content .wx-audio-right .wx-audio-progrees {
74 | height: 2px;
75 | width: calc(100% - 4px);
76 | position: relative;
77 | }
78 | .wx-audio-content .wx-audio-right .wx-audio-progrees .wx-progrees-detail {
79 | height: 100%;
80 | width: 100%;
81 | background: #ebebeb;
82 | position: relative;
83 | cursor: pointer;
84 | }
85 | .wx-audio-content .wx-audio-right .wx-audio-progrees .wx-progrees-detail .wx-voice-p {
86 | width: 0%;
87 | position: absolute;
88 | top: 0;
89 | bottom: 0;
90 | left: 0;
91 | right: 0;
92 | overflow: hidden;
93 | background: #09bb07;
94 | z-index: 2;
95 | }
96 | .wx-audio-content .wx-audio-right .wx-audio-progrees .wx-progrees-detail .wx-buffer-p {
97 | width: 0%;
98 | position: absolute;
99 | top: 0;
100 | bottom: 0;
101 | left: 0;
102 | right: 0;
103 | overflow: hidden;
104 | background: #d9d9d9;
105 | z-index: 1;
106 | }
107 | .wx-audio-content .wx-audio-right .wx-audio-progrees .wx-progrees-detail .wx-loading {
108 | display: none;
109 | position: absolute;
110 | top: 0;
111 | bottom: 0;
112 | left: 0;
113 | right: 0;
114 | overflow: hidden;
115 | }
116 | .wx-audio-content .wx-audio-right .wx-audio-progrees .wx-progrees-detail .wx-loading .wx-loading-wrapper {
117 | position: absolute;
118 | top: 0;
119 | bottom: 0;
120 | left: 0;
121 | animation: slidein 6s linear infinite normal;
122 | -webkit-animation: slidein 6s linear infinite normal;
123 | -moz-animation: slidein 6s linear infinite normal;
124 | width: 200%;
125 | max-width: none!important;
126 | background-image: -owg-repeating-linear-gradient(-15deg, #d9d9d9, #d9d9d9 2px, #ebebeb 2px, #ebebeb 4px);
127 | background-image: -webkit-repeating-linear-gradient(-15deg, #d9d9d9, #d9d9d9 2px, #ebebeb 2px, #ebebeb 4px);
128 | background-image: -moz-repeating-linear-gradient(-15deg, #d9d9d9, #d9d9d9 2px, #ebebeb 2px, #ebebeb 4px);
129 | background-image: -o-repeating-linear-gradient(-15deg, #d9d9d9, #d9d9d9 2px, #ebebeb 2px, #ebebeb 4px);
130 | background-image: repeating-linear-gradient(-15deg, #d9d9d9, #d9d9d9 2px, #ebebeb 2px, #ebebeb 4px);
131 | }
132 | .wx-audio-content .wx-audio-right .wx-audio-progrees .wx-audio-origin {
133 | width: 6px;
134 | height: 6px;
135 | margin-top: -3px;
136 | margin-left: 0;
137 | border-radius: 50%;
138 | -moz-border-radius: 50%;
139 | -webkit-border-radius: 50%;
140 | background-color: #09bb07;
141 | position: absolute;
142 | left: 0;
143 | top: 50%;
144 | z-index: 2;
145 | }
146 | .wx-audio-content .wx-audio-right .wx-audio-progrees .wx-audio-origin:before {
147 | content: " ";
148 | display: block;
149 | position: absolute;
150 | width: 24px;
151 | height: 24px;
152 | border-radius: 50%;
153 | -moz-border-radius: 50%;
154 | -webkit-border-radius: 50%;
155 | background-image: radial-gradient(rgba(9, 187, 7, 0.3) 20%, transparent 40%);
156 | top: 50%;
157 | margin-top: -12px;
158 | margin-left: -9px;
159 | cursor: pointer;
160 | outline: 0;
161 | -webkit-tap-highlight-color: rgba(0, 0, 0, 0);
162 | }
163 | .wx-audio-content .wx-audio-right .wx-audio-time {
164 | width: 100%;
165 | padding-top: 6px;
166 | height: auto;
167 | display: flex;
168 | align-items: center;
169 | justify-content: space-between;
170 | overflow: hidden;
171 | }
172 | .wx-audio-content .wx-audio-right .wx-audio-time span {
173 | font-size: 12px;
174 | color: #8c8c8c;
175 | }
176 |
--------------------------------------------------------------------------------
/lib/wx-audio.js:
--------------------------------------------------------------------------------
1 | var WxAudio=function(i){var t={};function e(A){if(t[A])return t[A].exports;var o=t[A]={i:A,l:!1,exports:{}};return i[A].call(o.exports,o,o.exports,e),o.l=!0,o.exports}return e.m=i,e.c=t,e.d=function(i,t,A){e.o(i,t)||Object.defineProperty(i,t,{enumerable:!0,get:A})},e.r=function(i){"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(i,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(i,"__esModule",{value:!0})},e.t=function(i,t){if(1&t&&(i=e(i)),8&t)return i;if(4&t&&"object"==typeof i&&i&&i.__esModule)return i;var A=Object.create(null);if(e.r(A),Object.defineProperty(A,"default",{enumerable:!0,value:i}),2&t&&"string"!=typeof i)for(var o in i)e.d(A,o,function(t){return i[t]}.bind(null,o));return A},e.n=function(i){var t=i&&i.__esModule?function(){return i.default}:function(){return i};return e.d(t,"a",t),t},e.o=function(i,t){return Object.prototype.hasOwnProperty.call(i,t)},e.p="",e(e.s=0)}([function(i,t,e){"use strict";Object.defineProperty(t,"__esModule",{value:!0});var A=function(){function i(i,t){for(var e=0;e0){for(var t=0,e=0;ei.durationT&&(t=i.durationT,i.showLoading(!1),console.log("缓冲完成"));var A=Math.floor(t/i.durationT*100);i.wxBufferP.style.width=A+"%"}var o=new Date;i.wxAudio.paused||(i.reduceTAfter=Date.parse(o)-Math.floor(1e3*i.currentT),i.reduceTAfter-i.reduceTBefore>1e3?i.showLoading(!0):i.showLoading(!1))},i.wxAudio.onended=function(){i.opt.ended()},i.wxAudio.ontimeupdate=function(){var t=new Date;i.isDrag||(i.currentT=i.wxAudio.currentTime,i.currentP=Number(i.wxAudio.currentTime/i.durationT*100),i.reduceTBefore=Date.parse(t)-Math.floor(1e3*i.currentT),i.currentP=i.currentP>100?100:i.currentP,i.wxVoiceP.style.width=i.currentP+"%",i.wxAudioOrigin.style.left=i.currentP+"%",i.wxAudioCurrent.innerText=i.formartTime(i.wxAudio.currentTime),i.showLoading(!1))},i.wxAudioStateImg.onclick=function(){i.audioPlayPause()},i.wxAudioOrigin.onmousedown=function(t){i.isDrag=!0;var e=(t||window.event).clientX,A=t.target.offsetLeft;i.maxProgressWidth=i.wxAudioDetail.offsetWidth,i.wxAudioC.onmousemove=function(t){if(i.isDrag){var o=(t||window.event).clientX;i.dragProgressTo=Math.min(i.maxProgressWidth,Math.max(0,A+(o-e))),i.updatePorgress()}},i.wxAudioC.onmouseup=function(){i.isDrag&&(i.isDrag=!1,i.wxAudio.currentTime=Math.floor(i.dragProgressTo/i.maxProgressWidth*i.durationT))},i.wxAudioC.onmouseleave=function(){i.isDrag&&(i.isDrag=!1,i.wxAudio.currentTime=Math.floor(i.dragProgressTo/i.maxProgressWidth*i.durationT))}},i.wxAudioOrigin.ontouchstart=function(t){i.isDrag=!0;var e=t||window.event,A=e.touches[0].clientX,o=e.target.offsetLeft;i.maxProgressWidth=i.wxAudioDetail.offsetWidth,i.wxAudioC.ontouchmove=function(t){if(i.isDrag){var e=(t||window.event).touches[0].clientX;i.dragProgressTo=Math.min(i.maxProgressWidth,Math.max(0,o+(e-A))),i.updatePorgress()}},i.wxAudioC.ontouchend=function(){i.isDrag&&(i.isDrag=!1,i.wxAudio.currentTime=Math.floor(i.dragProgressTo/i.maxProgressWidth*i.durationT))}},i.wxAudioDetail.onclick=function(t){var e=(t||window.event).layerX,A=i.wxAudioDetail.offsetWidth;i.wxAudio.currentTime=Math.floor(e/A*i.durationT)}}},{key:"isWeiXin",value:function(){var i=window.navigator.userAgent.toLowerCase();return"micromessenger"===String(i.match(/MicroMessenger/i))}},{key:"updatePorgress",value:function(){this.wxAudioOrigin.style.left=this.dragProgressTo+"px",this.wxVoiceP.style.width=this.dragProgressTo+"px";var i=Math.floor(this.dragProgressTo/this.maxProgressWidth*this.durationT);this.wxAudioCurrent.innerText=this.formartTime(i)}},{key:"formartTime",value:function(i){var t=function(i){return(i=i.toString())[1]?i:"0"+i},e=Math.floor(i/60),A=Math.floor(i%60);return t(e)+":"+t(A)}}]),i}();t.default=r},function(i,t){i.exports=""},function(i,t){i.exports=""},function(i,t){}]).default;
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "@dw/wx-audio",
3 | "version": "1.0.7",
4 | "description": "wx-audio",
5 | "main": "src/lib/wx-audio.js",
6 | "scripts": {
7 | "test": "echo \"Error: no test specified\" && exit 1",
8 | "build": "webpack --mode production",
9 | "build:publish": "webpack --mode production && npm version patch && npm publish",
10 | "dev": "webpack-dev-server --mode development --progress --colors",
11 | "watch": "webpack --progress --colors --watch",
12 | "publish": "npm publish --access=public"
13 | },
14 | "repository": {
15 | "type": "git",
16 | "url": "https://www.github.com/ifmiss/wx-audio.js"
17 | },
18 | "files": [
19 | "lib/",
20 | "src/lib",
21 | "src/assets",
22 | "README.md"
23 | ],
24 | "keywords": [
25 | "audio",
26 | "wx",
27 | "we-audio",
28 | "wx-audio",
29 | "music"
30 | ],
31 | "author": "daiwei",
32 | "license": "ISC",
33 | "devDependencies": {
34 | "@dw/webpack-prompt-plugin": "^1.0.7",
35 | "axios": "^0.18.0",
36 | "babel-core": "^6.26.0",
37 | "babel-loader": "^7.1.4",
38 | "babel-polyfill": "^6.26.0",
39 | "babel-preset-env": "^1.6.1",
40 | "babel-preset-latest": "^6.24.1",
41 | "css-loader": "^0.28.10",
42 | "extract-text-webpack-plugin": "^4.0.0-beta.0",
43 | "html-webpack-plugin": "^3.0.6",
44 | "less": "^3.9.0",
45 | "less-loader": "^4.1.0",
46 | "style-loader": "^0.20.3",
47 | "url-loader": "^1.0.1",
48 | "url-parse": "^1.4.3",
49 | "webpack": "^4.1.1",
50 | "webpack-cli": "^3.3.1",
51 | "webpack-dev-server": "^3.1.1"
52 | }
53 | }
54 |
--------------------------------------------------------------------------------
/src/assets/pause.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/IFmiss/wx-audio/8d99e43cbd2ab039983564045b0e3542518fac1d/src/assets/pause.png
--------------------------------------------------------------------------------
/src/assets/playing.gif:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/IFmiss/wx-audio/8d99e43cbd2ab039983564045b0e3542518fac1d/src/assets/playing.gif
--------------------------------------------------------------------------------
/src/index.js:
--------------------------------------------------------------------------------
1 | import './style.less'
2 | import WxAudio from './lib/wx-audio.js';
3 |
4 | import axios from 'axios'
5 | let index = 0
6 | let wx, music
7 |
8 | // axios.get('http://www.daiwei.org/vue/server/music.php?inAjax=1&do=playlist&id=2179377798').then((res) => {
9 | // music = res.data.playlist.tracks
10 | // index = Math.floor(Math.random() * music.length)
11 | // axios.get('http://www.daiwei.org/vue/server/music.php?inAjax=1&do=musicInfo&id=' + music[index].id).then((res) => {
12 | // wx = new WxAudio ({
13 | // ele: '.wx-audio',
14 | // title: music[index].name,
15 | // disc: music[index].ar[0].name,
16 | // src: res.data.data[0].url,
17 | // width: '320px',
18 | // ended: function () {
19 | // index = Math.floor(Math.random() * music.length)
20 | // axios.get('http://www.daiwei.org/vue/server/music.php?inAjax=1&do=musicInfo&id=' + music[index].id).then((res) => {
21 | // console.log(music[index].ar[0].name)
22 | // var src = res.data.data[0].url
23 | // var title = music[index].name
24 | // var disc = music[index].ar[0].name
25 | // wx.audioCut(src, title, disc)
26 | // }, (err) => {
27 | // console.log(err)
28 | // })
29 | // }
30 | // })
31 | // })
32 | // }, (err) => {
33 | // console.log(err)
34 | // })
35 | wx = new WxAudio ({
36 | ele: '.wx-audio',
37 | title: '111111',
38 | disc: '2222',
39 | src: 'http://m10.music.126.net/20190426110819/bdb6a025449d78de141a2c31a2330484/ymusic/cd1c/7825/d746/0dc64d2be36bfe20f55a401318b6080a.mp3',
40 | width: '320px',
41 | autoplay: true
42 | })
43 |
44 | document.getElementById('play').onclick = function () {
45 | wx.audioPlay()
46 | }
47 |
48 | document.getElementById('pause').onclick = function () {
49 | wx.audioPause()
50 | }
51 |
52 | document.getElementById('cut').onclick = function () {
53 | index = Math.floor(Math.random() * music.length)
54 | axios.get('http://www.daiwei.org/vue/server/music.php?inAjax=1&do=musicInfo&id=' + music[index].id).then((res) => {
55 | console.log(music[index].ar[0].name)
56 | var src = res.data.data[0].url
57 | var title = music[index].name
58 | var disc = music[index].ar[0].name
59 | wx.audioCut(src, title, disc)
60 | }, (err) => {
61 | console.log(err)
62 | })
63 | }
64 |
--------------------------------------------------------------------------------
/src/lib/wx-audio.js:
--------------------------------------------------------------------------------
1 | import './wx-audio.less'
2 | const playImg = require('./../assets/playing.gif')
3 | const pauseImg = require('./../assets/pause.png')
4 | export default class WxAudio {
5 | constructor (tpl) {
6 | let defaultOptions = {
7 | ele: null,
8 | width: '320px',
9 | title: '这是一个测试title',
10 | src: '',
11 | disc: '这是一个测试disc',
12 | autoplay: false,
13 | // loop为true的时候不执行ended事件
14 | loop: true,
15 | ended: function () {}
16 | }
17 | this.opt = Object.assign({}, defaultOptions, tpl)
18 | // 判断传进来的是DOM还是字符串
19 | if (typeof this.opt.ele === 'string') {
20 | this.opt.ele = document.querySelector(this.opt.ele)
21 | }
22 | if (!this.opt.ele) return
23 |
24 | this.loading = false
25 | this.isDrag = false
26 | this.isplaying = false
27 | this.durationT = 0
28 | this.currentT = 0
29 | this.currentP = 0
30 | this.maxProgressWidth = 0
31 | this.dragProgressTo = 0
32 |
33 | // 通过时间戳与当前时间的差值来判断是否需要加载
34 | this.reduceTBefore = 0 // 时间戳与当前时间的差值 (初始化)
35 | this.reduceTAfter = 0 // 时间戳与当前时间的差值 (执行中)
36 |
37 | this.initDom();
38 | }
39 |
40 | // 初始化元素
41 | initDom () {
42 | // content
43 | this.wxAudioC = document.createElement('div')
44 | this.wxAudioC.className = 'wx-audio-content'
45 | this.wxAudioC.style.width = this.opt.width
46 | this.opt.ele.appendChild(this.wxAudioC)
47 |
48 | // audio
49 | this.wxAudio = document.createElement('audio')
50 | this.wxAudio.className = 'wx-audio-content'
51 | this.wxAudio.src = this.opt.src
52 | if (this.opt.loop) this.wxAudio.setAttribute('loop', this.opt.loop)
53 | this.wxAudioC.appendChild(this.wxAudio)
54 |
55 | // left
56 | this.wxAudioL = document.createElement('div')
57 | this.wxAudioL.className = 'wx-audio-left'
58 | this.wxAudioC.appendChild(this.wxAudioL)
59 |
60 | // left image
61 | this.wxAudioStateImg = document.createElement('img')
62 | this.wxAudioStateImg.className = 'wx-audio-state'
63 | this.wxAudioStateImg.src = pauseImg
64 | this.wxAudioL.appendChild(this.wxAudioStateImg)
65 |
66 | // right
67 | this.wxAudioR = document.createElement('div')
68 | this.wxAudioR.className = 'wx-audio-right'
69 | this.wxAudioC.appendChild(this.wxAudioR)
70 |
71 | // right title
72 | this.wxAudioT = document.createElement('p')
73 | this.wxAudioT.className = 'wx-audio-title'
74 | this.wxAudioT.innerText = this.opt.title
75 | this.wxAudioR.appendChild(this.wxAudioT)
76 |
77 | // right disc
78 | this.wxAudioD = document.createElement('p')
79 | this.wxAudioD.className = 'wx-audio-disc'
80 | this.wxAudioD.innerText = this.opt.disc
81 | this.wxAudioR.appendChild(this.wxAudioD)
82 |
83 | // right progrees
84 | this.wxAudioP = document.createElement('div')
85 | this.wxAudioP.className = 'wx-audio-progrees'
86 | this.wxAudioR.appendChild(this.wxAudioP)
87 |
88 | // right progress detail
89 | this.wxAudioDetail = document.createElement('div')
90 | this.wxAudioDetail.className = 'wx-progrees-detail'
91 | this.wxAudioP.appendChild(this.wxAudioDetail)
92 |
93 | // voice p
94 | this.wxVoiceP = document.createElement('span')
95 | this.wxVoiceP.className = 'wx-voice-p'
96 | this.wxAudioDetail.appendChild(this.wxVoiceP)
97 |
98 | // buffer p
99 | this.wxBufferP = document.createElement('span')
100 | this.wxBufferP.className = 'wx-buffer-p'
101 | this.wxAudioDetail.appendChild(this.wxBufferP)
102 |
103 | // loading p
104 | this.wxLoading = document.createElement('span')
105 | this.wxLoading.className = 'wx-loading'
106 | this.wxAudioDetail.appendChild(this.wxLoading)
107 |
108 | // laoding wrapper
109 | this.wxLoadingWrapper = document.createElement('span')
110 | this.wxLoadingWrapper.className = 'wx-loading-wrapper'
111 | this.wxLoading.appendChild(this.wxLoadingWrapper)
112 |
113 | // origin
114 | this.wxAudioOrigin = document.createElement('div')
115 | this.wxAudioOrigin.className = 'wx-audio-origin'
116 | this.wxAudioP.appendChild(this.wxAudioOrigin)
117 |
118 | // 音乐时间信息
119 | this.wxAudioTime = document.createElement('div')
120 | this.wxAudioTime.className = 'wx-audio-time'
121 | this.wxAudioR.appendChild(this.wxAudioTime)
122 |
123 | // currentT
124 | this.wxAudioCurrent = document.createElement('span')
125 | this.wxAudioCurrent.className = 'current-t'
126 | this.wxAudioCurrent.innerText = '00:00'
127 | this.wxAudioTime.appendChild(this.wxAudioCurrent)
128 |
129 | // durationT
130 | this.wxAudioDuration = document.createElement('span')
131 | this.wxAudioDuration.className = 'duration-t'
132 | this.wxAudioDuration.innerText = '00:00'
133 | this.wxAudioTime.appendChild(this.wxAudioDuration)
134 |
135 | this.initAudioEvent();
136 |
137 | if (this.opt.autoplay) {
138 | document.addEventListener('WeixinJSBridgeReady', () => {
139 | this.audioPlay()
140 | })
141 | }
142 | }
143 |
144 | // 播放
145 | audioPlay () {
146 | this.wxAudio.play()
147 | this.isPlaying = true
148 | }
149 |
150 | //
151 | audioPause () {
152 | this.wxAudio.pause();
153 | this.isPlaying = false
154 | }
155 |
156 | audioPlayPause () {
157 | if (this.isPlaying) {
158 | this.audioPause();
159 | } else {
160 | this.audioPlay();
161 | }
162 | }
163 |
164 | audioCut (src, title, disc) {
165 | this.wxAudio.src = src
166 | this.wxAudioT.innerText = title
167 | this.wxAudioD.innerText = disc
168 | this.durationT = 0
169 | this.currentT = 0
170 | this.currentP = 0
171 | this.dragProgressTo = 0
172 | // 初始化 wxAudioCurrent 的文案
173 | this.wxAudioCurrent.innerText = '00:00'
174 | this.wxAudioOrigin.style.left = '0px'
175 | this.wxVoiceP.style.width = '0px'
176 | this.audioPlay()
177 | }
178 |
179 | showLoading (bool) {
180 | this.loading = bool || false
181 | if (this.loading) {
182 | this.wxLoading.style.display = 'block'
183 | } else {
184 | this.wxLoading.style.display = 'none'
185 | }
186 | }
187 |
188 | initAudioEvent () {
189 | var _this = this
190 | // 音频事件
191 | _this.wxAudio.onplaying = function () {
192 | var date = new Date ()
193 | _this.isPlaying = true
194 | _this.reduceTBefore = Date.parse(date) - Math.floor(_this.wxAudio.currentTime * 1000)
195 | _this.wxAudioStateImg.src = playImg
196 | },
197 | _this.wxAudio.onpause = function () {
198 | _this.isPlaying = false
199 | _this.showLoading(false)
200 | _this.wxAudioStateImg.src = pauseImg
201 | },
202 | _this.wxAudio.onloadedmetadata = function () {
203 | _this.durationT = _this.wxAudio.duration
204 | // 初始化视频时间
205 | _this.wxAudioDuration.innerText = _this.formartTime(_this.wxAudio.duration)
206 | },
207 | _this.wxAudio.onwaiting = function () {
208 | if(!_this.wxAudio.paused) {
209 | _this.showLoading(true)
210 | }
211 | },
212 | _this.wxAudio.onprogress = function () {
213 | if(_this.wxAudio.buffered.length > 0) {
214 | var bufferedT = 0
215 | for (var i = 0; i < _this.wxAudio.buffered.length; i++) {
216 | bufferedT += _this.wxAudio.buffered.end(i) - _this.wxAudio.buffered.start(i)
217 | if(bufferedT > _this.durationT) {
218 | bufferedT = _this.durationT
219 | _this.showLoading(false)
220 | console.log('缓冲完成')
221 | }
222 | }
223 | var bufferedP = Math.floor((bufferedT / _this.durationT) * 100)
224 | _this.wxBufferP.style.width = bufferedP + '%'
225 | }
226 |
227 | // ===========================
228 | var date = new Date ()
229 | if(!_this.wxAudio.paused) {
230 | _this.reduceTAfter = Date.parse(date) - Math.floor(_this.currentT * 1000)
231 | if(_this.reduceTAfter - _this.reduceTBefore > 1000) {
232 | _this.showLoading(true)
233 | } else {
234 | _this.showLoading(false)
235 | }
236 | } else {
237 | return
238 | }
239 | },
240 | // 播放结束事件
241 | _this.wxAudio.onended = function () {
242 | _this.opt.ended()
243 | }
244 | // 绑定进度条
245 | _this.wxAudio.ontimeupdate = function () {
246 | var date = new Date ()
247 | if (!_this.isDrag) {
248 | _this.currentT = _this.wxAudio.currentTime
249 | _this.currentP = Number((_this.wxAudio.currentTime / _this.durationT) * 100)
250 | _this.reduceTBefore = Date.parse(date) - Math.floor(_this.currentT * 1000)
251 | _this.currentP = _this.currentP > 100 ? 100 : _this.currentP
252 | _this.wxVoiceP.style.width = _this.currentP + '%'
253 | _this.wxAudioOrigin.style.left = _this.currentP + '%'
254 | // 更改时间进度
255 | _this.wxAudioCurrent.innerText = _this.formartTime(_this.wxAudio.currentTime)
256 | _this.showLoading(false)
257 | }
258 | },
259 | // 页面点击事件
260 | _this.wxAudioStateImg.onclick = function () {
261 | _this.audioPlayPause()
262 | }
263 |
264 | _this.wxAudioOrigin.onmousedown = function (event) {
265 | _this.isDrag = true
266 | var e = event || window.event
267 | var x = e.clientX
268 | var l = event.target.offsetLeft
269 | _this.maxProgressWidth = _this.wxAudioDetail.offsetWidth
270 | _this.wxAudioC.onmousemove = function (event) {
271 | if (_this.isDrag) {
272 | var e = event || window.event
273 | var thisX = e.clientX
274 | _this.dragProgressTo = Math.min(_this.maxProgressWidth, Math.max(0, l + (thisX - x)))
275 | // update Time
276 | _this.updatePorgress()
277 | }
278 | }
279 | _this.wxAudioC.onmouseup = function () {
280 | if (_this.isDrag) {
281 | _this.isDrag = false
282 | _this.wxAudio.currentTime = Math.floor(_this.dragProgressTo / _this.maxProgressWidth * _this.durationT)
283 | } else {
284 | return
285 | }
286 | }
287 |
288 | _this.wxAudioC.onmouseleave = function () {
289 | if (_this.isDrag) {
290 | _this.isDrag = false
291 | _this.wxAudio.currentTime = Math.floor(_this.dragProgressTo / _this.maxProgressWidth * _this.durationT)
292 | } else {
293 | return
294 | }
295 | }
296 | }
297 |
298 | _this.wxAudioOrigin.ontouchstart = function (event) {
299 | _this.isDrag = true
300 | var e = event || window.event
301 | var x = e.touches[0].clientX
302 | var l = e.target.offsetLeft
303 |
304 | _this.maxProgressWidth = _this.wxAudioDetail.offsetWidth
305 |
306 | _this.wxAudioC.ontouchmove = function (event) {
307 | if (_this.isDrag) {
308 | var e = event || window.event
309 | var thisX = e.touches[0].clientX
310 | _this.dragProgressTo = Math.min(_this.maxProgressWidth, Math.max(0, l + (thisX - x)))
311 | _this.updatePorgress()
312 | }
313 | },
314 | _this.wxAudioC.ontouchend = function () {
315 | if (_this.isDrag) {
316 | _this.isDrag = false
317 | _this.wxAudio.currentTime = Math.floor(_this.dragProgressTo / _this.maxProgressWidth * _this.durationT)
318 | } else {
319 | return
320 | }
321 | }
322 | }
323 |
324 | _this.wxAudioDetail.onclick = function (event) {
325 | var e = event || window.event
326 | var l = e.layerX
327 | var w = _this.wxAudioDetail.offsetWidth
328 | _this.wxAudio.currentTime = Math.floor(l / w * _this.durationT)
329 | }
330 | }
331 |
332 | isWeiXin () {
333 | const uaLower = window.navigator.userAgent.toLowerCase()
334 | return String(uaLower.match(/MicroMessenger/i)) === 'micromessenger'
335 | }
336 |
337 | updatePorgress () {
338 | this.wxAudioOrigin.style.left = this.dragProgressTo + 'px'
339 | this.wxVoiceP.style.width = this.dragProgressTo + 'px'
340 | var currentTime = Math.floor(this.dragProgressTo / this.maxProgressWidth * this.durationT)
341 | // this.wxAudio.currentTime = currentTime
342 | this.wxAudioCurrent.innerText = this.formartTime(currentTime)
343 | // this.wxAudio.currentTime = Math.floor(this.dragProgressTo / this.maxProgressWidth * this.durationT)
344 | }
345 |
346 | formartTime (seconds) {
347 | var formatNumber = function (n) {
348 | n = n.toString()
349 | return n[1] ? n : '0' + n
350 | }
351 | var m = Math.floor(seconds / 60);
352 | var s = Math.floor(seconds % 60);
353 | return formatNumber(m) + ":" + formatNumber(s);
354 | }
355 | }
356 |
--------------------------------------------------------------------------------
/src/lib/wx-audio.less:
--------------------------------------------------------------------------------
1 | @keyframes slidein {
2 | 0% {
3 | -webkit-transform: translateX(-50%);
4 | -moz-transform: translateX(-50%);
5 | -ms-transform: translateX(-50%);
6 | -o-transform: translateX(-50%);
7 | transform: translateX(-50%);
8 | -webkit-transform: translateX(-50%);
9 | -moz-transform: translateX(-50%);
10 | -ms-transform: translateX(-50%);
11 | -o-transform: translateX(-50%);
12 | transform: translateX(-50%); }
13 | 100% {
14 | -webkit-transform: translateX(0);
15 | -moz-transform: translateX(0);
16 | -ms-transform: translateX(0);
17 | -o-transform: translateX(0);
18 | transform: translateX(0);
19 | -webkit-transform: translateX(0);
20 | -moz-transform: translateX(0);
21 | -ms-transform: translateX(0);
22 | -o-transform: translateX(0);
23 | transform: translateX(0); } }
24 | .wx-audio-content{
25 | user-select: none;
26 | width: 100%;
27 | display: flex;
28 | align-items: center;
29 | justify-content: center;
30 | min-height: auto;
31 | padding: 8px 12px;
32 | border: 1px solid #efefef;
33 | margin: 0 auto;
34 | font-size: 0;
35 | box-sizing: border-box;
36 | font-family: 'PingFangSC-Regular';
37 | -webkit-tap-highlight-color:transparent;
38 | background:#fdfdfd;
39 | p{
40 | margin: 0;
41 | }
42 | .wx-audio-info{
43 | display: none;
44 | }
45 | .wx-audio-left{
46 | display: inline-block;
47 | width:42px;
48 | height: 100%;
49 | .wx-audio-state{
50 | width: 100%;
51 | height: 100%;
52 | }
53 | }
54 | .wx-audio-right{
55 | display: inline-block;
56 | width: calc(100% - 42px);
57 | font-size: 16px;
58 | box-sizing: border-box;
59 | padding-left:10px;
60 | .wx-audio-title{
61 | padding-bottom: 6px;
62 | width: 100%;
63 | font-size: 16px;
64 | font-weight: 400;
65 | text-overflow: ellipsis;
66 | overflow: hidden;
67 | white-space: nowrap;
68 | }
69 | .wx-audio-disc{
70 | width: 100%;
71 | text-overflow: ellipsis;
72 | overflow: hidden;
73 | white-space: nowrap;
74 | padding-bottom: 10px;
75 | font-size: 12px;
76 | color: #8c8c8c;
77 | }
78 | .wx-audio-progrees{
79 | height: 2px;
80 | width: calc(100% - 4px);
81 | position:relative;
82 | .wx-progrees-detail{
83 | height: 100%;
84 | width: 100%;
85 | background:#ebebeb;
86 | position:relative;
87 | cursor: pointer;
88 | .wx-voice-p{
89 | width: 0%;
90 | position: absolute;
91 | top: 0;
92 | bottom: 0;
93 | left: 0;
94 | right: 0;
95 | overflow: hidden;
96 | background:#09bb07;
97 | z-index: 2;
98 | }
99 | .wx-buffer-p{
100 | width: 0%;
101 | position: absolute;
102 | top: 0;
103 | bottom: 0;
104 | left: 0;
105 | right: 0;
106 | overflow: hidden;
107 | background:#d9d9d9;
108 | z-index: 1;
109 | }
110 | .wx-loading{
111 | display: none;
112 | position: absolute;
113 | top: 0;
114 | bottom: 0;
115 | left: 0;
116 | right: 0;
117 | overflow: hidden;
118 | .wx-loading-wrapper{
119 | position: absolute;
120 | top: 0;
121 | bottom: 0;
122 | left: 0;
123 | animation: slidein 6s linear infinite normal;
124 | -webkit-animation: slidein 6s linear infinite normal;
125 | -moz-animation: slidein 6s linear infinite normal;
126 | width: 200%;
127 | max-width: none!important;
128 | background-image: -owg-repeating-linear-gradient(-15deg, #d9d9d9, #d9d9d9 2px, #ebebeb 2px, #ebebeb 4px);
129 | background-image: -webkit-repeating-linear-gradient(-15deg, #d9d9d9, #d9d9d9 2px, #ebebeb 2px, #ebebeb 4px);
130 | background-image: -moz-repeating-linear-gradient(-15deg, #d9d9d9, #d9d9d9 2px, #ebebeb 2px, #ebebeb 4px);
131 | background-image: -o-repeating-linear-gradient(-15deg, #d9d9d9, #d9d9d9 2px, #ebebeb 2px, #ebebeb 4px);
132 | background-image: repeating-linear-gradient(-15deg, #d9d9d9, #d9d9d9 2px, #ebebeb 2px, #ebebeb 4px);
133 | }
134 | }
135 | }
136 | .wx-audio-origin{
137 | width: 6px;
138 | height: 6px;
139 | margin-top: -3px;
140 | margin-left: 0;
141 | border-radius: 50%;
142 | -moz-border-radius: 50%;
143 | -webkit-border-radius: 50%;
144 | background-color: #09bb07;
145 | position: absolute;
146 | left: 0;
147 | top: 50%;
148 | z-index: 2;
149 | &:before{
150 | content: " ";
151 | display: block;
152 | // z-index: 2;
153 | position: absolute;
154 | width: 24px;
155 | height: 24px;
156 | border-radius: 50%;
157 | -moz-border-radius: 50%;
158 | -webkit-border-radius: 50%;
159 | // background-color: rgba(9,187,7,0.3);
160 | background-image: radial-gradient(rgba(9,187,7,0.3) 20%, transparent 40%);
161 | top: 50%;
162 | margin-top: -12px;
163 | margin-left: -9px;
164 | cursor: pointer;
165 | outline: 0;
166 | -webkit-tap-highlight-color: rgba(0,0,0,0);
167 | }
168 | }
169 | }
170 | .wx-audio-time{
171 | width: 100%;
172 | padding-top: 6px;
173 | height: auto;
174 | display: flex;
175 | align-items: center;
176 | justify-content: space-between;
177 | overflow: hidden;
178 | span{
179 | font-size: 12px;
180 | color: #8c8c8c;
181 | }
182 | }
183 | }
184 | }
185 |
--------------------------------------------------------------------------------
/src/style.less:
--------------------------------------------------------------------------------
1 | body,html{
2 | margin: 0;
3 | padding: 0;
4 | height:100%;
5 | background: #fff;
6 | }
7 |
8 | .colortext-lt (@color1, @color2) {
9 | position: relative;
10 | background: -webkit-linear-gradient(left top, @color1 , @color2); /* Safari 5.1 - 6.0 */
11 | background: -o-linear-gradient(bottom right, @color1, @color2); /* Opera 11.1 - 12.0 */
12 | background: -moz-linear-gradient(bottom right, @color1, @color2); /* Firefox 3.6 - 15 */
13 | background: linear-gradient(to bottom right, @color1 , @color2); /* 标准的语法 */
14 | -webkit-background-clip: text;
15 | -webkit-text-fill-color: transparent;
16 | }
17 |
18 | body{
19 | // background: red;
20 | display: flex;
21 | width: 100%;
22 | height: 100%;
23 | align-items: center;
24 | flex-direction: column;
25 | justify-content: center;
26 | .title{
27 | font-size: 42px;
28 | display: inline-block;
29 | .colortext-lt(rgba(50, 219, 154, 0.856), rgba(209, 163, 47, 0.808));
30 | margin-top: 0;
31 | }
32 | .disc{
33 | font-size: 14px;
34 | color: #f1f1f1;
35 | }
36 | .github-info{
37 | margin-top: 40px;
38 | text-align: center;
39 | display: flex;
40 | justify-content: center;
41 | span{
42 | color: #ffffff;
43 | font: Arial,"Helvetica Neue",Helvetica,"Microsoft Yahei","Hiragino Sans GB","Heiti SC","WenQuanYi Micro Hei",sans-serif;
44 | font-size:14px;
45 | }
46 | }
47 | .btn{
48 | width: 320px;
49 | line-height: 40px;
50 | background: #999;
51 | cursor: pointer;
52 | color: #fff;
53 | text-align: center;
54 | margin-top: 30px;
55 | }
56 | }
--------------------------------------------------------------------------------
/tea.yaml:
--------------------------------------------------------------------------------
1 | # https://tea.xyz/what-is-this-file
2 | ---
3 | version: 1.0.0
4 | codeOwners:
5 | - '0x18fb62c5c736405ece7E8e7a2F01D084fFa77B6c'
6 | quorum: 1
7 |
--------------------------------------------------------------------------------
/webpack.config.js:
--------------------------------------------------------------------------------
1 | const webpack = require('webpack'); // 用于访问内置插件
2 | const path = require('path');
3 |
4 | const HtmlWebpackPlugin = require('html-webpack-plugin');
5 | const ExtractTextPlugin = require("extract-text-webpack-plugin");
6 | const WebpackPromptPlugin = require("@dw/webpack-prompt-plugin")
7 |
8 | const extractSass = new ExtractTextPlugin({
9 | filename: "wx-audio.css",
10 | disable: process.env.NODE_ENV === "development"
11 | });
12 |
13 | const resolve = function (dir) {
14 | return path.resolve(__dirname, dir);
15 | }
16 |
17 | module.exports = {
18 | entry: {
19 | index: 'src/lib/wx-audio.js'
20 | // index: './src/index.js'
21 | },
22 | output: {
23 | path: path.resolve(__dirname, 'lib'),
24 | filename: 'wx-audio.js',
25 | libraryTarget: 'var',
26 | library: 'WxAudio',
27 | libraryExport: 'default'
28 | },
29 | module: {
30 | rules: [
31 | {
32 | test: /\.css$/,
33 | use: ExtractTextPlugin.extract({
34 | fallback:"style-loader",
35 | use:["css-loader"]
36 | })
37 | },
38 | {
39 | test: /\.less$/,
40 | use: ExtractTextPlugin.extract({
41 | fallback:"style-loader",
42 | use:[
43 | {
44 | loader: 'css-loader'
45 | },
46 | {
47 | loader: 'less-loader'
48 | }
49 | ]
50 | })
51 | },
52 | {
53 | test: /\.(ttf|eot|svg|woff|woff2)$/,
54 | use: [
55 | {
56 | loader: 'url-loader'
57 | }
58 | ]
59 | },
60 | {
61 | test: /\.js$/,
62 | exclude: /(node_modules)/,
63 | use: {
64 | loader: 'babel-loader',
65 | options: {
66 | presets: ["env"]
67 | }
68 | }
69 | },
70 | {
71 | test: /\.(png|jpg|gif)$/,
72 | use: [
73 | {
74 | loader: 'url-loader',
75 | options: {
76 | limit: 8192
77 | }
78 | }
79 | ]
80 | },
81 | ]
82 | },
83 | plugins: [
84 | // new HtmlWebpackPlugin ({
85 | // filename: 'index.html',
86 | // template: 'index.html',
87 | // inject: true
88 | // }),
89 | extractSass,
90 | new WebpackPromptPlugin()
91 | ],
92 | devServer: {
93 | // 当使用 HTML5 History API 时,任意的 404 响应都可能需要被替代为 index.html。通过传入以下启用:
94 | contentBase: "./",
95 | host: '0.0.0.0',
96 | // 端口号
97 | port: 1997,
98 | //当有编译器错误或警告时,在浏览器中显示全屏覆盖。默认禁用。如果您只想显示编译器错误:
99 | noInfo: true,
100 | // 配置端口号
101 | overlay: true,
102 | },
103 | resolve: {
104 | alias: {
105 | 'src': resolve('src'),
106 | 'commonjs': resolve('src/commonjs'),
107 | 'scss': resolve('src/scss'),
108 | 'stylus': resolve('src/stylus'),
109 | 'script': resolve('src/script'),
110 | 'static': resolve('static'),
111 | }
112 | },
113 | optimization: {
114 | splitChunks: {
115 | chunks: "all",
116 | minSize: 30000,
117 | minChunks: 3,
118 | maxAsyncRequests: 5,
119 | maxInitialRequests: 3,
120 | name: true,
121 | cacheGroups: {
122 | default: {
123 | minChunks: 2,
124 | priority: -20,
125 | reuseExistingChunk: true,
126 | },
127 | vendors: {
128 | test: /[\\/]node_modules[\\/]/,
129 | priority: -10
130 | }
131 | }
132 | }
133 | }
134 | }
135 |
--------------------------------------------------------------------------------