From ff545c5d559b6771086fa059d874a8b639e77562 Mon Sep 17 00:00:00 2001 From: Michael Dzjaparidze Date: Sun, 10 Dec 2017 12:41:28 +0100 Subject: [PATCH] Update readme --- README.md | 215 ++++++++++++++++++++++++---------------------- src/waveplayer.js | 10 +-- 2 files changed, 116 insertions(+), 109 deletions(-) diff --git a/README.md b/README.md index bb1eb264..2078656d 100755 --- a/README.md +++ b/README.md @@ -1,6 +1,6 @@ # waveplayer.js -An HTML5 based audio player with a waveform view. Inspired by [wavesurfer.js](http://wavesurfer-js.org/) and [waveform.js](http://www.waveformjs.org/), although the implementation is quite different from both. +An HTML5 based audio player with a waveform view. Inspired by [wavesurfer.js](http://wavesurfer-js.org/) and [waveform.js](http://www.waveformjs.org/), although the implementation is quite different from both. ![Screenshot](waveform.png?raw=true "Example") @@ -8,162 +8,169 @@ Author: Michael Dzjaparidze License: MIT -### Browser Support & Other Requirements -waveplayer.js relies on new ES6 features such as promises and generators, and hence, will only work on modern browsers. waveplayer.js is tested to work on Chrome, Firefox and Opera (no Internet Explorer yet). Safari 8 doesn't seem to support generators yet. +## Browser Support & Other Requirements +waveplayer.js is tested to work on Chrome, Firefox and Opera (no Internet Explorer yet). -In order to minimize waveform drawing and audio file loading times, it is necessary to supply an URL to a JSON file representing the waveform of the audio file numerically. This JSON file should be named identically to the corresponding audio file and should exist at the same location. If this JSON file does not exist, an error will be raised. +In order to minimize waveform drawing and audio file loading times, it is necessary to supply an URL to a JSON file representing the waveform of the audio file numerically. This JSON file should be named identically to the corresponding audio file and should exist at the same location. An error will be thrown if this JSON file does not exist. There exist a number of tools for extracting waveform data in JSON format from an audio file; [wav2json](https://github.com/beschulz/wav2json) or [py-wav2json](https://github.com/michaeldzjap/py-wav2json) can be used for instance. It is enough to provide a single array of floating point values. If the JSON contains an associative array, only the first entree will be used. -### Example 1: load an audio file and play it +## Installation +This package is available through [npm](https://www.npmjs.com/): +``` +npm install --save waveplayer +``` +The package is also still available through [bower](https://bower.io/), although this will be faded out eventually +``` +bower install waveplayer.js +``` +After that the package may be directly included in a ` +``` +or may be imported (ES6) in your JavaScript file: +```javascript +import WavePlayer from 'waveplayer'; +``` +## Examples + +### Load an audio file and play it Create a waveplayer.js instance and pass in some (optional) options: ```javascript -var waveplayer = new WavePlayer({ - container: '#waveform', - barWidth: 4, - barGap: 1, - height: 128 +import WavePlayer from 'waveplayer'; + +var wavePlayer = new WavePlayer({ + container: '#waveform', + barWidth: 4, + barGap: 1, + height: 128 }); ``` Load an audio file from a URL and start playback when loading has finished: ```javascript -waveplayer.load("audio/fm2.mp3") - .then(function() { - waveplayer.play(); - }); +wavePlayer.load('url-to-some-audio-file.mp3') + .then(() => waveplayer.play()); ``` -### Example 2: schedule a playlist and play it through from start to finish - -Load some audio files from a URL and start playback when loading has finished: +### Schedule a playlist and play it through from start to finish +Load some audio files from an URL and start playback when loading has finished: ```javascript -waveplayer.schedulePlaylist({ - urls: ["audio/fm2.mp3", "audio/fm3.mp3", "audio/fm7.mp3", "audio/fm8.mp3", "audio/fm13.mp3"], - onStart: function() { // optional function to be evaluated when first track has finished loading - waveplayer.play(); - }, - onChange: function() { // optional function to be evaluated between playback of consecutive tracks - console.log("next track will be loaded"); - }, - onEnd: function() { // optional function to be evaluated when the playlist reached its end - console.log("playlist is finished"); - } +wavePlayer.createPlaylist( + ["1.mp3", "2.mp3", "3.mp3", "4.mp3", "5.mp3"] +); + +wavePlayer.on('waveplayer:playlist:ready', me => { + me.play(); }); ``` +View the code for a full playlist example [here](/example/src/playlist.js) -View the code for a full playlist example [here](/examples/demo.js) - -### waveplayer.js Options +## waveplayer.js Options | option | type | default | description | | --- | --- | --- | --- | | `barGap` | integer | `1` | Gap between bars in pixels. | | `barWidth` | integer | `4` | Width of a bar in pixels. | | `container` | mixed | `null` | **Required**: CSS-selector or HTML-element for displaying the waveform. | -| `data` | array | `null` | An array of floating point values representing waveform amplitudes. Note that it is not required to set this explicitly as an option, although it ultimately needs to be supplied by the user somehow (see the [`load`](#waveplayerjs-methods) method for more information). | | `height` | integer | `128` | The height of the waveform in pixels. | | `interact` | boolean | `true` | Enables/disables mouse interaction with the waveform view. This may be changed at any point after creation. | | `progressColor` | string | `#31708f` | The fill color of the waveform bars that have been played back so far. | | `responsive` | boolean | `true` | If set to true, the width of the waveform view adapts to the width of the container element. | | `waveColor` | string | `#428bca` | The fill color of the waveform bars that have not been played back so far. | +| `width` | integer | 512 | The width of the waveform in pixels (only relevant when the `responsive` option is set to `false`). | -### waveplayer.js Methods - +## waveplayer.js Methods * `cancelPlaylist()` - Cancels the currently active playlist (will also stop the audio instantly). - + Cancels the currently active playlist (will also stop the audio instantly). +* `createPlaylist(urls)` + + Creates a new playlist (it will cancel the currently active playlist if there is one). + + **Arguments**: + + `urls` is an array of valid audio file URL's. + + `options` (**optional**) is an object with the following keys: + + - `autoPlay`: A boolean that indicates if the playlist should start playing automatically after it was created. * `destroy()` - - Stops playback, cancels any running playlists and unsubscribes from any events that are listened for. -* `ended()` - - Returns a Promise which may be used to perform an action when a track finishes. + Stops playback, cancels any running playlists and unsubscribes from any events that are listened for. +* `load(url)` + + Load an audio file from a URL and return a Promise which may be used to perform an action when the audio file has finished loading. -* `load()` + **Arguments**: - Load an audio file from a URL and return a Promise which may be used to perform an action when the audio file has finished loading. - -* `interact(bool)` + `url` is a valid URL to an audio file. +* `interact` or `interact = bool` - Setter or getter for enabling/disabling mouse interaction with the waveform view. - - **Arguments**: - `bool` is an optional boolean value for determining whether mouse interaction with the waveform view should be enabled or disabled. + Setter or getter for enabling/disabling mouse interaction with the waveform view. + **Arguments**: + + `bool` is a boolean value for determining whether mouse interaction with the waveform view should be enabled or disabled. * `isPlaying()` - Returns a boolean indicating whether audio is currently playing or not. - + Returns a boolean indicating whether audio is currently playing or not. * `on(topic, fn)` - Subscribe to an event with a custom event handler. - - **Arguments**: - `topic` is a string denoting a valid event name to which to subscribe. - - `fn()` is the callback thay will be evaluated every time the event is fired. - + Subscribe to an event with a custom event handler. + + **Arguments**: + + `topic` is a string denoting a valid event name to which to subscribe. + + `fn()` is the callback that will be evaluated every time the event is fired. * `pause()` - Pause the currently playing audio file. - + Pause the currently playing audio file. * `play()` - Start playback of currently loaded audio file. - -* `responsive(bool)` + Start playback of currently loaded audio file. +* `responsive` or `responsive = bool` - Setter or getter for enabling/disabling responsive mode for the waveform view. If set, the width of the waveform view will adapt to the width of its parent container element. - - **Arguments**: - `bool` is an optional boolean value for determining whether to switch between a responsive or fixed design of the waveform view. + Setter or getter for enabling/disabling responsive mode for the waveform view. If set, the width of the waveform view will adapt to the width of its parent container element. -* `schedulePlaylist(options)` + **Arguments**: - Schedules a new playlist. - - **Arguments**: - `options` is an associative array holding the following objects: - - `urls`: Array of URL's. **Required** - - `onStart()`: Optional callback function that will be evaluated when first track in playlist has finished loading. - - `onChange()`: Optional callback function that will be evaluated between playback of consecutive tracks. - - `onEnd()`: Optional callback function that will be evaluated when the playlist reaches its end. + `bool` is a boolean value for determining whether to switch between a responsive or fixed design of the waveform view. +* `skipTo(seconds)` -* `skipToSec(sec)` + Set the current time in seconds from which to playback audio. - Set the current time in seconds from which to playback audio. - - **Arguments**: - `sec` is a float value denoting the current time from which to playback the audio file. + **Arguments**: + `seconds` is a float value denoting the current time from which to playback the audio file. * `un(topic, fn)` - - Unsubscribe a particular callback or from the whole event all together. - - **Arguments**: - `topic` is a string denoting a valid event name from which to unsubscribe. - - `fn()` is an optional callback from which to unsubscribe. - -* `volume(val)` - - Set or get the audio playback volume, range: `[0 - 1]`. - - **Arguments**: - `val` sets the audio playback volume to this value (if provided, otherwise current set value is returned). - -### waveplayer.js Events - -* `canplay` - Identical to the corresponding [Media event](https://developer.mozilla.org/en-US/docs/Web/Guide/Events/Media_events). Sent when enough data is available so that the media can be played, at least for a couple of frames. -* `ended` - Identical to the corresponding [Media event](https://developer.mozilla.org/en-US/docs/Web/Guide/Events/Media_events). Sent when playback of an audio file is completed. -* `playlistended` - Sent when the playlist has finished playing all the queued audio. -* `playlistqueued` - Sent when the the first audio file in the playlist has finished loading. -* `waveclicked` - Sent when the user clicks somewhere on the waveform. Callback will receive a float denoting the progress `[0..1]`. -* `waveclickplay` - Sent when the user initiates playback by clicking on the waveform view with the mouse. + + Unsubscribe a particular callback or from the whole event all together. + + **Arguments**: + + `topic` is a string denoting a valid event name from which to unsubscribe. + + `fn()` (**optional**) is a callback from which to unsubscribe. + +* `volume` or `volume = value` + + Set or get the audio playback volume, range: `[0 - 1]`. + + **Arguments**: + + `value` sets the audio playback volume to this value. + +## waveplayer.js Events + +* `waveplayer:canplay` - Identical to the corresponding [Media event](https://developer.mozilla.org/en-US/docs/Web/Guide/Events/Media_events). Fired when enough data is available so that the media can be played, at least for a couple of frames. Callback will receive the waveplayer instance. +* `waveplayer:initialized` - Fired when the waveform view has been constructed and the HTML audio element has been created. Callback will receive the waveplayer instance. +* `waveplayer:skipped` - Fired when the audio file was skipped forward or backward in time. Callback will receive the waveplayer instance and the new position of the playback header (in seconds). +* `waveplayer:playlist:finished` - Fired when the playlist has finished playing all the queued audio. Callback will receive the waveplayer instance. +* `waveplayer:playlist:next` - Fired when the next track in the playlist has been loaded and is about to be played. Callback will receive the waveplayer instance and an object with track information. +* `waveplayer:playlist:ready` - Fired when the the first audio file in the playlist has finished loading. Callback will receive the waveplayer instance. diff --git a/src/waveplayer.js b/src/waveplayer.js index e1badf52..d5dfcfdd 100644 --- a/src/waveplayer.js +++ b/src/waveplayer.js @@ -62,7 +62,7 @@ class WavePlayer { Promise.all([ this._initializeAudioElm(), this._initializeWaveViewInteraction() - ]).then(() => WavePlayer._mediator.fire('waveplayer:initialized')); + ]).then(() => WavePlayer._mediator.fire('waveplayer:initialized', this)); } /************************ @@ -225,14 +225,14 @@ class WavePlayer { * * @param {Array} urls * @param {Object} options - * @return {void} + * @return {Playlist} */ createPlaylist(urls, options = {}) { - // Cancel current playlist before starting a new one this.cancelPlaylist(); - this._playlist = new Playlist(this, urls, options); + + return this._playlist; } /** @@ -319,7 +319,7 @@ class WavePlayer { this._waveView.container.appendChild(this._audioElm); this._canplayHandler = () => { - WavePlayer._mediator.fire('waveplayer:canplay'); + WavePlayer._mediator.fire('waveplayer:canplay', this); resolve('waveplayer:canplay'); }; this._audioElm.addEventListener('canplay', this._canplayHandler.bind(this));