org.deepsymmetry.beatlink.data.WaveformDetail Maven / Gradle / Ivy
package org.deepsymmetry.beatlink.data;
import org.deepsymmetry.beatlink.Util;
import org.deepsymmetry.beatlink.dbserver.BinaryField;
import org.deepsymmetry.beatlink.dbserver.Message;
import javax.swing.*;
import java.nio.ByteBuffer;
/**
* Gives a detail view of the audio content of a track, and offers a Swing component for rendering that view
* as part of a user interface, along with annotations showing the current playback position, beats, and cue points,
* if the appropriate metadata is available.
*
* @author James Elliott
*/
public class WaveformDetail {
/**
* The number of bytes at the start of the waveform data which do not seem to be valid or used.
*/
@SuppressWarnings("WeakerAccess")
public static final int LEADING_JUNK_BYTES = 19;
/**
* The unique identifier that was used to request this waveform detail.
*/
@SuppressWarnings("WeakerAccess")
public final DataReference dataReference;
/**
* The message holding the detail as it was read over the network. This can be used to analyze fields
* that have not yet been reliably understood, and is also used for storing the cue list in a cache file.
*/
@SuppressWarnings("WeakerAccess")
public final Message rawMessage;
/**
* Get the raw bytes of the waveform detail data
*
* @return the bytes from which the detail can be drawn, as described in Section 5.8 of the
* Packet Analysis document.
*/
@SuppressWarnings("WeakerAccess")
public ByteBuffer getData() {
return ((BinaryField) rawMessage.arguments.get(3)).getValue();
}
/**
* Count the half-frames of waveform available.
*
* @return the number of half-frames that make up the track, ignoring the leading junk bytes
*/
@SuppressWarnings("WeakerAccess")
public int getFrameCount() {
return getData().remaining() - LEADING_JUNK_BYTES;
}
/**
* Determine how long the track plays, in milliseconds. This provides a more accurate value than the track
* metadata, which is accurate only to the second, because we know how many half-frames (1/150 of a second)
* the track is composed of.
*
* @return the number of milliseconds it will take to play all half-frames that make up the track
*/
public long getTotalTime() {
return Util.halfFrameToTime(getFrameCount());
}
/**
* Create a standard Swing component which can be added to a user interface that will draw this waveform detail,
* optionally including annotations like the current playback position and minute markers (if you supply
* {@link TrackMetadata} so the total length can be determined), and cue markers (if you also supply a
* {@link CueList}). The playback position can be
*
* @param metadata Information about the track whose waveform we are drawing, so we can translate times into
* positions
* @param beatGrid The locations of all the beats in the track, so they can be drawn
*
* @return the component which will draw the annotated waveform preview
*/
public JComponent createViewComponent(TrackMetadata metadata, BeatGrid beatGrid) {
return new WaveformDetailComponent(this, metadata, beatGrid);
}
/**
* Constructor when reading from the network or a cache file.
*
* @param reference the unique database reference that was used to request this waveform detail
* @param message the response that contains the preview
*/
@SuppressWarnings("WeakerAccess")
public WaveformDetail(DataReference reference, Message message) {
dataReference = reference;
rawMessage = message;
}
@Override
public String toString() {
return "WaveformDetail[dataReference=" + dataReference + ", size:" + getData().remaining() + "]";
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy