Code Monkey home page Code Monkey logo

peaks.js's Introduction

Peaks.js Build Status

A browser based audio waveform visualisation frontend component from BBC R&D.

Peaks is a modular frontend component designed for the display of and interaction with audio waveform material in the browser.

Peaks is developed by BBC R&D to allow users to make accurate clippings of audio data over a timeline in browser, using a backend API that serves the waveform data.

Peaks utilizes HTML5 canvas technology to display waveform data at different zoom levels and provides some basic convenience methods for interacting with waveforms and creating time-based visual sections for denoting content to be clipped or for reference, eg: distinguishing music from speech or identifying different music tracks.

More informations about the Waveform project.

Installation

  • npm: npm install --save peaks.js
  • bower: bower install --save peaks.js
  • old school: download zip file

Using Peaks.js in your own project

Peaks can be included in any web page by following these steps:

  1. include it your web page
  2. include a media element and its waveform data file
  3. initialise Peaks
<div id="peaks-container"></div>
<audio>
  <source src="test_data/sample.mp3" type="audio/mpeg">
  <source src="test_data/sample.ogg" type="audio/ogg">
</audio>
<script src="bower_components/requirejs/require.js" data-main="app.js"></script>

Start using require.js

AMD modules are working out of the box without any optimiser.

// in app.js
// configure peaks path
requirejs.config({
  paths: {
    peaks: 'bower_components/peaks.js/src/main',
    EventEmitter: 'bower_components/eventemitter2/lib/eventemitter2',
    Kinetic: 'bower_components/kineticjs/kinetic',
    'waveform-data': 'bower_components/waveform-data/dist/waveform-data.min'
  }
});

// requires it
require(['peaks'], function (Peaks) {
  var p = Peaks.init({
    container: document.querySelector('#peaks-container'),
    mediaElement: document.querySelector('audio'),
    dataUri: 'test_data/sample.json'
  });

  p.on('segments.ready', function(){
    // do something when segments are ready to be displayed
  });
});

A working examples is provided in index.html.

Start using vanilla JavaScript

<script src="http://wzrd.in/standalone/peaks.js"></script>
<script>
(function(Peaks){
    var p = Peaks.init({});
})(peaks.js);
</script>

Notice: this examples uses Browserify CDN: save it as a local copy or build your own.

Please refer to the Building Peaks section to learn more about how to build Peaks for development or production purpose.

Generate waveform data

Peaks uses waveform data files produced by audiowaveform. These can be generated in either binary (.dat) or JSON format. Binary format is preferred because of the smaller file size, but this is only compatible with browsers that support Typed Arrays.

You should also use the -b 8 option when generating waveform data files, as Peaks does not currently support 16-bit waveform data files, and also to minimise file size.

To generate a binary waveform data file:

audiowaveform -i sample.mp3 -o sample.dat -b 8

To generate a JSON format waveform data file:

audiowaveform -i sample.mp3 -o sample.json -b 8

Refer to audiowaveform(1) for full details of the available command line options.

Web Audio based waveforms

Since 0.3.0, Peaks can rely on Web Audio to generate waveforms. Which means you would not have to pre-generate a dat or json beforehand.

To do so, skip the dataUri option and make sure your browser is compatible with Web Audio.

var p = Peaks.init({
  container: document.querySelector('#peaks-container'),
  mediaElement: document.querySelector('audio')
});

p.on('segments.ready', function(){
  // do something when segments are ready to be displayed
});

Notice: be aware it can be CPU intensive if your audio file has a long duration.

Configuration

The available options for configuration of the viewer are as follows:

var options = {
  /** REQUIRED OPTIONS **/
  // Containing element
  container: document.getElementById('peaks-container'),

  // HTML5 Media element containing an audio track
  mediaElement: document.querySelector('audio'),

  /** Optional config with defaults **/
  // URI to waveform data file in binary or JSON
  dataUri: {
    arraybuffer: '../test_data/sample.dat',
    json: '../test_data/sample.json',
  },

  // async logging function
  logger: console.error.bind(console),

  // default height of the waveform canvases in pixels
  height: 200,

  // Array of zoom levels in samples per pixel (big >> small)
  zoomLevels: [512, 1024, 2048, 4096],

  // Bind keyboard controls
  keyboard: false,

  // Keyboard nudge increment in seconds (left arrow/right arrow)
  nudgeIncrement: 0.01,

  // Colour for the in marker of segments
  inMarkerColor: '#a0a0a0',

  // Colour for the out marker of segments
  outMarkerColor: '#a0a0a0',

  // Colour for the zoomed in waveform
  zoomWaveformColor: 'rgba(0, 225, 128, 1)',

  // Colour for the overview waveform
  overviewWaveformColor: 'rgba(0,0,0,0.2)',

  // Colour for the overview waveform rectangle that shows what the zoom view shows
  overviewHighlightRectangleColor: 'grey',

  // Colour for segments on the waveform
  segmentColor: 'rgba(255, 161, 39, 1)',

  // Colour of the play head
  playheadColor: 'rgba(0, 0, 0, 1)',

  // Colour of the play head text
  playheadTextColor: '#aaa',

  // the color of a point marker
  pointMarkerColor: '#FF0000',

  // Colour of the axis gridlines
  axisGridlineColor: '#ccc',

  // Colour of the axis labels
  axisLabelColor: '#aaa',

  // Random colour per segment (overrides segmentColor)
  randomizeSegmentColor: true,

  // Zoom view adapter to use. Valid adapters are: 'animated' (default) and 'static'
  zoomAdapter: 'animated',

  // Array of initial segment objects with startTime and
  // endTime in seconds and a boolean for editable.
  // See below.
  segments: [{
    startTime: 120,
    endTime: 140,
    editable: true,
    color: "#ff0000",
    labelText: "My label"
  },
  {
    startTime: 220,
    endTime: 240,
    editable: false,
    color: "#00ff00",
    labelText: "My Second label"
  }]
}

Segment Format

Segments provided from peaks.js use the following format:

[{
   // Assigned colour of the segment
  color: "rgba(123, 2, 61, 1)",
   // Editable state of the segment
  editable: true,
   // End time in seconds of the segment
  endTime: 588.986667,
   // Unique ID of the segment
  id: "segment0",
   // Kinetics.js Element group of segment canvas objects for overview waveform
  overview: Kinetic.Group,
   // End time in seconds of the segment
  startTime: 578.986667,
   // Kinetics.js Element group of segment canvas objects for overview waveform
  zoom: Kinetic.Group
}]

Advanced configuration

The marker and label Kinetic.js objects may be overridden to give the segment markers or label your own custom appearance (see main.js / waveform.mixins.js, Kinetic Polygon Example and Kinetic Text Example) :

{
  segmentInMarker: mixins.defaultInMarker(p.options),
  segmentOutMarker: mixins.defaultOutMarker(p.options),
  segmentLabelDraw: mixins.defaultSegmentLabelDraw(p.options)
}

API

The top level peaks object exposes a factory to create new peaks instances.

peaks.init(options)

Start a instance of peaks with the assigned options. It enables you do deal with several instances of Peaks within a single page with one or several configurations.

var peaksInstance = peaks.init({});
var secondPeaksInstance = peaks.init({});

Time API

instance.time.getCurrentTime()

Returns currently selected time in seconds (convenience method interchangeable with mediaElement.currentTime).

var instance = peaks.init({});

console.log(instance.time.getCurrentTime()); // -> 0

instance.time.setCurrentTime(timeInSeconds)

Sets the media element selected time in seconds.

var instance = peaks.init({});

instance.time.setCurrentTime(5.85);
console.log(instance.time.getCurrentTime()); // -> 5.85

Zoom API

instance.zoom.zoomOut()

Zoom in the waveform zoom view by one level.

var instance = peaks.init({, zoomLevels: [512, 1024, 2048, 4096] });

instance.zoom.zoomOut(); // zoom level is now 1024

instance.zoom.zoomIn()

Zoom in the waveform zoom view by one level.

var instance = peaks.init({, zoomLevels: [512, 1024, 2048, 4096] });

instance.zoom.zoomIn(); // zoom level is still 512

instance.zoom.zoomOut(); // zoom level is now 1024
instance.zoom.zoomIn(); // zoom level is now 512 again

instance.zoom.setZoom(indexInZoomArray)

Set the zoom level to the element in the options.zoomLevels array at index indexInZoomArray.

var instance = peaks.init({, zoomLevels: [512, 1024, 2048, 4096] });

instance.zoom.setZoom(3); // zoom level is now 4096

instance.zoom.getZoom()

Return the current zoom level.

var instance = peaks.init({, zoomLevels: [512, 1024, 2048, 4096] });

instance.zoom.zoomOut();
console.log(instance.zoom.getZoom()); // -> 1

Segments API

Segments give the ability to visually tag timed portions of a media object. This is a great way to provide visual cues to your users.

instance.segments.add(startTime, endTime, editable, color, labelText)

instance.segments.add(segment[])

Add a segment to the waveform timeline with starting time startTime (seconds), ending time endTime (seconds) and set whether the segment is user editable with editable (boolean, defaults to false).

var instance = peaks.init({});

instance.segments.add(0, 10.5); //adds a 0 to 10.5 seconds non-editable segment with a random color

Alternatively, provide an array of segment objects as shown above in the config options as the first and only argument to add all those segments at once.

var instance = peaks.init({});

instance.segments.add([
  {
    startTime: 0,
    endTime: 10.5,
    labelText: '0 to 10 seconds non-editable demo segment'
  },
  {
    startTime: 3.14,
    endTime: 4.2,
    color: '#666'
  }
]);

instance.segments.getSegments()

Returns an array of objects representing all displayed segments present on the timeline in the segment format.

instance.segments.removeByTime(timeStart, timeEnd)

Remove any segment which start at timeStart, and which optionally finish at timeEnd.

The returned value corresponds to the number of deleted segments.

var instance = peaks.init({});

instance.segments.add([{ startTime: 10, endTime: 12 }, { startTime: 10, endTime: 20 }]);

// remove both segments as they start at `10`
instance.segments.removeByTime(10);

// remove only the first segment
instance.segments.removeByTime(10, 12);

instance.segments.removeAll()

Remove all the segments from the waveforms.

var instance = peaks.init({});

instance.segments.removeAll();

Events

Peaks emit events to enable you to extend its behaviour according to your needs.

General

                       | Arguments

---------------------------|---------------- error | Error err

Media / User interactions

                       | Arguments

---------------------------|---------------- segments.ready | N/A zoomview_resized | N/A user_seek.overview | float time user_seek.zoomview | float time user_scrub.overview | float time

Waveforms

                       | Arguments

---------------------------|---------------- zoom.update | float currentZoomLevel, float previousZoomLevel

Segments / Points

                       | Arguments

---------------------------|---------------- segments.ready | N/A points.ready | N/A segment.dragged | Segment segment

Building Peaks

You might want to build a minified standalone version of Peaks, to test a contribution or to run additional tests. The project bundles everything you need to do so.

Prerequisite

git clone https://github.com/bbcrd/peaks.js.git
cd peaks.js
npm install

Building

This command will produce a minified standalone version of Peaks. It will indeed be UMD compatible, so as you can continue to use it with AMD or CommonJS module loaders, or even as Vanilla JavaScript.

npm run build

The output of the build will be located as peaks.min.js, alongside its associated Source Maps.

Live Demo

This command will serve a working version of Peaks to reflect your local updates.

npm start

Then open http://localhost:9000 in a Web browser.

Testing

npm test should work for simple one time testing.

If you are developing and want to repeatedly run tests in a browser on your machine simply launch npm run test-watch.

License

See COPYING

This project includes sample audio from the radio show Desert Island Discs, used under the terms of the Creative Commons 3.0 Unported License.

Credits

Copyright 2015, British Broadcasting Corporation.

peaks.js's People

Contributors

thom4parisot avatar ziggythehamster avatar chainlink avatar johvet avatar chrisn avatar un-chein-andalou avatar bitwit avatar samstarling avatar yosiat avatar thinkpunk avatar

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    🖖 Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. 📊📈🎉

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google ❤️ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.