├── .babelrc ├── .gitignore..git ├── LICENSE ├── README.md ├── index.html ├── package.json ├── public ├── Consolation.mp3 ├── illenium_cover.jpg ├── miro_cover.jpg ├── with_u_newVersion_4_FINAL.mp3 ├── without_you.jpg └── without_you.mp3 ├── react-audio-component.sketch ├── server.js ├── src ├── actions │ └── index.js ├── assets │ └── default_cover.txt ├── audio │ └── ReactAudio.js ├── components │ ├── ArtistInfo.css │ ├── ArtistInfo.js │ ├── AudioControlButton.css │ ├── AudioControlButton.js │ ├── Utilities.css │ ├── Utilities.js │ └── svg │ │ ├── HeartIcon.js │ │ ├── NextIcon.js │ │ ├── PauseIcon.js │ │ ├── PlayIcon.js │ │ ├── PreviousIcon.js │ │ ├── RepeatIcon.js │ │ ├── ShareIcon.js │ │ └── SpeakerIcon.js ├── constants │ └── ActionTypes.js ├── containers │ ├── App.css │ └── App.js ├── css │ ├── ReactSlider.css │ ├── index.css │ └── variables.css ├── index.js ├── reducers │ ├── audio.js │ └── index.js ├── store │ ├── configureStore.js │ └── songs.json └── utils │ └── detection.js ├── webpack.hot.js └── webpack.prod.js /.babelrc: -------------------------------------------------------------------------------- 1 | { 2 | "presets" : ["es2015", "react", "stage-0"], 3 | "plugins": ["transform-decorators-legacy"] 4 | } 5 | -------------------------------------------------------------------------------- /.gitignore..git: -------------------------------------------------------------------------------- 1 | # Logs 2 | logs 3 | *.log 4 | npm-debug.log* 5 | 6 | # Runtime data 7 | pids 8 | *.pid 9 | *.seed 10 | 11 | # Directory for instrumented libs generated by jscoverage/JSCover 12 | lib-cov 13 | 14 | # Coverage directory used by tools like istanbul 15 | coverage 16 | 17 | # Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files) 18 | .grunt 19 | 20 | # node-waf configuration 21 | .lock-wscript 22 | 23 | # Compiled binary addons (http://nodejs.org/api/addons.html) 24 | build/Release 25 | 26 | # Dependency directory 27 | node_modules 28 | 29 | # Optional npm cache directory 30 | .npm 31 | 32 | # Optional REPL history 33 | .node_repl_history 34 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | The MIT License (MIT) 2 | 3 | Copyright (c) 2016 J.S. Leonard 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # React Audio Component Example Project 2 | 3 | A beautiful example project demonstrating how to build an audio player in React. Desktop optimized though it works on mobile down to iPhone 5 resolutions. 4 | 5 | ## The Player 6 | 7 | ![Audio Player](http://leonardsouza.com/public/images/portfolio/audio_player.jpg) 8 | 9 | Demo [here](http://leonardsouza.com/react-audio-example.html). Check it out! 10 | 11 | ### Quick Start 12 | 13 | `npm install` 14 | 15 | `npm start` 16 | 17 | `open localhost:3000` 18 | 19 | 20 | _Rest of README forthcoming_ 21 | -------------------------------------------------------------------------------- /index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | React Audio Player 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 |
15 | 16 | 17 | 18 | 19 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "react-audio-component", 3 | "version": "1.0.0", 4 | "description": "A beautiful audio player built with React", 5 | "main": "index.js", 6 | "scripts": { 7 | "start": "node server.js", 8 | "deploy": "webpack -p --config webpack.prod.js --output-file audio_player.js && mv ./public/audio_player.js ../portfolio/jslauthor.github.io/public", 9 | "test": "echo \"Error: no test specified\" && exit 1" 10 | }, 11 | "repository": { 12 | "type": "git", 13 | "url": "git+https://github.com/jslauthor/react-audio-component.git" 14 | }, 15 | "keywords": [ 16 | "audio", 17 | "html5", 18 | "react" 19 | ], 20 | "author": "J.S. Leonard", 21 | "license": "ISC", 22 | "bugs": { 23 | "url": "https://github.com/jslauthor/react-audio-component/issues" 24 | }, 25 | "homepage": "https://github.com/jslauthor/react-audio-component#readme", 26 | "devDependencies": { 27 | "autoprefixer": "^6.3.6", 28 | "css-loader": "^0.23.1", 29 | "postcss-bem": "^0.4.1", 30 | "postcss-color-function": "^2.0.1", 31 | "postcss-custom-media": "^5.0.1", 32 | "postcss-import": "^8.1.2", 33 | "postcss-loader": "^0.9.1", 34 | "postcss-mixins": "^4.0.2", 35 | "postcss-nested": "^1.0.0", 36 | "postcss-sass-colors": "0.0.2", 37 | "postcss-simple-vars": "^2.0.0", 38 | "precss": "^1.4.0", 39 | "react-hot-loader": "^1.3.0", 40 | "style-loader": "^0.13.1", 41 | "webpack": "^1.13.1", 42 | "webpack-dev-server": "^1.14.1" 43 | }, 44 | "dependencies": { 45 | "babel-loader": "^6.2.4", 46 | "babel-plugin-transform-decorators-legacy": "^1.3.4", 47 | "babel-preset-es2015": "^6.9.0", 48 | "babel-preset-react": "^6.5.0", 49 | "babel-preset-stage-0": "^6.5.0", 50 | "imports-loader": "^0.6.5", 51 | "json-loader": "^0.5.4", 52 | "lodash": "^4.13.1", 53 | "marked": "^0.3.5", 54 | "raf": "^3.2.0", 55 | "raw-loader": "^0.5.1", 56 | "rc-progress": "^1.0.4", 57 | "rc-slider": "^3.7.1", 58 | "rc-tooltip": "^3.3.2", 59 | "react": "^15.1.0", 60 | "react-addons-css-transition-group": "^15.1.0", 61 | "react-dom": "^15.1.0", 62 | "react-marquee": "^0.1.0", 63 | "react-redux": "^4.4.5", 64 | "react-slider": "^0.6.1", 65 | "react-swipeable": "^3.5.0", 66 | "redux": "^3.5.2" 67 | } 68 | } 69 | -------------------------------------------------------------------------------- /public/Consolation.mp3: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jslauthor/react-audio-component/6d8589f573c9ccabfc91f1fcf100c758fedf8e9c/public/Consolation.mp3 -------------------------------------------------------------------------------- /public/illenium_cover.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jslauthor/react-audio-component/6d8589f573c9ccabfc91f1fcf100c758fedf8e9c/public/illenium_cover.jpg -------------------------------------------------------------------------------- /public/miro_cover.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jslauthor/react-audio-component/6d8589f573c9ccabfc91f1fcf100c758fedf8e9c/public/miro_cover.jpg -------------------------------------------------------------------------------- /public/with_u_newVersion_4_FINAL.mp3: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jslauthor/react-audio-component/6d8589f573c9ccabfc91f1fcf100c758fedf8e9c/public/with_u_newVersion_4_FINAL.mp3 -------------------------------------------------------------------------------- /public/without_you.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jslauthor/react-audio-component/6d8589f573c9ccabfc91f1fcf100c758fedf8e9c/public/without_you.jpg -------------------------------------------------------------------------------- /public/without_you.mp3: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jslauthor/react-audio-component/6d8589f573c9ccabfc91f1fcf100c758fedf8e9c/public/without_you.mp3 -------------------------------------------------------------------------------- /react-audio-component.sketch: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jslauthor/react-audio-component/6d8589f573c9ccabfc91f1fcf100c758fedf8e9c/react-audio-component.sketch -------------------------------------------------------------------------------- /server.js: -------------------------------------------------------------------------------- 1 | 2 | var webpack = require('webpack'); 3 | var WebpackDevServer = require('webpack-dev-server'); 4 | var config = require('./webpack.hot'); 5 | 6 | new WebpackDevServer(webpack(config), { 7 | publicPath: config.output.publicPath, 8 | hot: true, 9 | historyApiFallback: true 10 | }).listen(3000, '0.0.0.0', function (err, result) { 11 | if (err) { 12 | return console.log(err); 13 | } 14 | 15 | console.log('Listening at http://0.0.0.0:3000/'); 16 | }); 17 | -------------------------------------------------------------------------------- /src/actions/index.js: -------------------------------------------------------------------------------- 1 | import * as types from '../constants/ActionTypes' 2 | import data from '../store/songs.json'; 3 | 4 | export function retrieveSongs() { 5 | // This could be written as a thunk or saga 6 | return { type: types.INITIALIZE, songs: data.songs } 7 | } 8 | 9 | export function play(audio) { 10 | if (audio.paused) 11 | audio.play(); 12 | else 13 | audio.pause(); 14 | 15 | return { type: types.PLAY, audio } 16 | } 17 | 18 | export function pause(audio) { 19 | audio.pause(); 20 | return { type: types.PAUSE, audio } 21 | } 22 | 23 | function resetAudio(audio) { 24 | // need to reset the song if it's the same file 25 | audio.currentTime = 0; 26 | const src = audio.src; 27 | audio.src = null; 28 | audio.src = src; 29 | } 30 | 31 | export function next(audio) { 32 | resetAudio(audio) 33 | return { type: types.NEXT, audio } 34 | } 35 | 36 | export function previous(audio) { 37 | resetAudio(audio) 38 | return { type: types.PREVIOUS, audio } 39 | } 40 | 41 | export function updateVolume(audio, volume) { 42 | audio.volume = volume/100; 43 | return { type: types.UPDATE_VOLUME, volume } 44 | } 45 | 46 | export function setTime(audio) { 47 | const percent = audio.currentTime / audio.duration; 48 | return { type: types.SET_TIME, audio } 49 | } 50 | 51 | export function setProgress(audio) { 52 | return { type: types.SET_PROGRESS, audio } 53 | } 54 | 55 | export function setError(audio) { 56 | return { type: types.ERROR, audio } 57 | } 58 | 59 | export function updatePosition(audio, percent) { 60 | audio.currentTime = percent * audio.duration; 61 | return { type: types.UPDATE_POSITION, audio } 62 | } 63 | 64 | export function toggleFavorite(id) { 65 | return { type: types.TOGGLE_FAVORITE, id } 66 | } 67 | 68 | export function toggleRepeat() { 69 | return { type: types.TOGGLE_REPEAT } 70 | } 71 | 72 | export function toggleLoop(audio) { 73 | audio.loop = !audio.loop; 74 | return { type: types.TOGGLE_LOOP, audio } 75 | } 76 | -------------------------------------------------------------------------------- /src/assets/default_cover.txt: -------------------------------------------------------------------------------- 1 |  2 | -------------------------------------------------------------------------------- /src/audio/ReactAudio.js: -------------------------------------------------------------------------------- 1 | import React from 'react'; 2 | import ReactDOM from 'react-dom'; 3 | import isFunction from 'lodash/isFunction'; 4 | import partialRight from 'lodash/partialRight'; 5 | import forEach from 'lodash/forEach'; 6 | 7 | class ReactAudio extends React.Component { 8 | 9 | static propTypes = { 10 | autoplay: React.PropTypes.bool, 11 | preload: React.PropTypes.bool, 12 | source: React.PropTypes.string, 13 | loop: React.PropTypes.bool, 14 | volume: React.PropTypes.number, 15 | onTimeupdate: React.PropTypes.func, 16 | onError: React.PropTypes.func, 17 | onProgress: React.PropTypes.func, 18 | onEnded: React.PropTypes.func 19 | }; 20 | 21 | static defaultProps = { 22 | autoplay: false, 23 | preload: true, 24 | source: "", 25 | loop: false, 26 | volume: .8, 27 | onTimeupdate: null, 28 | onError: null, 29 | onProgress: null, 30 | onEnded: null 31 | }; 32 | 33 | constructor(props) { 34 | super(props) 35 | 36 | this.state = { 37 | listeners: [] 38 | }; 39 | } 40 | 41 | get audio() { 42 | if (!this.refs) 43 | return {}; 44 | 45 | return ReactDOM.findDOMNode(this.refs.audio); 46 | } 47 | 48 | set audio(a) {} 49 | 50 | handler(e, func) { 51 | if (isFunction(func)) { 52 | func(e); 53 | } 54 | } 55 | 56 | addListener = (event, func) => { 57 | var audio = ReactDOM.findDOMNode(this.refs.audio); 58 | audio.addEventListener(event, partialRight(this.handler, func)); 59 | this.state.listeners.push({event: event, func: func}); 60 | } 61 | 62 | removeAllListeners = () => { 63 | var audio = ReactDOM.findDOMNode(this.refs.audio); 64 | forEach(this.state.listeners, (obj) => { 65 | audio.removeEventListener(obj.event, obj.func); 66 | }) 67 | this.state.listeners = []; 68 | } 69 | 70 | componentDidMount() { 71 | this.addListener('timeupdate', this.props.onTimeupdate); 72 | this.addListener('progress', this.props.onProgress); 73 | this.addListener('error', this.props.onError); 74 | this.addListener('ended', this.props.onEnded); 75 | this.addListener('loadeddata', this.props.onLoadedData); 76 | } 77 | 78 | componentWillUnmount() { 79 | this.removeAllListeners(); 80 | } 81 | 82 | componentWillReceiveProps(nextProps) { 83 | if (nextProps.autoplay === true && this.props.autoplay === false) { 84 | this.audio.play(); 85 | } 86 | } 87 | 88 | togglePlay = () => { 89 | if (this.audio.paused) 90 | this.audio.play(); 91 | else 92 | this.audio.pause(); 93 | } 94 | 95 | setPlaybackPercent(percent) { 96 | this.audio.currentTime = percent * this.audio.duration; 97 | } 98 | 99 | changeCurrentTimeBy = (amount) => { 100 | this.audio.currentTime += amount; 101 | } 102 | 103 | setVolume = (percent) => { 104 | this.audio.volume = percent; 105 | } 106 | 107 | render() { 108 | return( 109 |