All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.drew.metadata.wav.WavRiffHandler Maven / Gradle / Ivy

package com.drew.metadata.wav;

import com.drew.imaging.riff.RiffHandler;
import com.drew.lang.ByteArrayReader;
import com.drew.lang.annotations.NotNull;
import com.drew.metadata.Metadata;
import com.drew.metadata.MetadataException;

import java.io.IOException;

/**
 * Implementation of {@link RiffHandler} specialising in Wav support.
 *
 * Extracts data from chunk/list types:
 *
 * 
    *
  • "INFO": artist, title, product, track number, date created, genre, comments, copyright, software, duration
  • *
  • "fmt ": format, channels, samples/second, bytes/second, block alignment, bits/sample
  • *
  • "data": duration
  • *
* * Sources: http://www.neurophys.wisc.edu/auditory/riff-format.txt * http://www-mmsp.ece.mcgill.ca/Documents/AudioFormats/WAVE/WAVE.html * http://wiki.audacityteam.org/wiki/WAV * * @author Payton Garland */ public class WavRiffHandler implements RiffHandler { @NotNull private final WavDirectory _directory; @NotNull private String _currentList = ""; public WavRiffHandler(@NotNull Metadata metadata) { _directory = new WavDirectory(); metadata.addDirectory(_directory); } public boolean shouldAcceptRiffIdentifier(@NotNull String identifier) { return identifier.equals(WavDirectory.FORMAT); } public boolean shouldAcceptChunk(@NotNull String fourCC) { return fourCC.equals(WavDirectory.CHUNK_FORMAT) || (_currentList.equals(WavDirectory.LIST_INFO) && WavDirectory._tagIntegerMap.containsKey(fourCC)) || fourCC.equals(WavDirectory.CHUNK_DATA); } @Override public boolean shouldAcceptList(@NotNull String fourCC) { if (fourCC.equals(WavDirectory.LIST_INFO)) { _currentList = WavDirectory.LIST_INFO; return true; } else { _currentList = ""; return false; } } public void processChunk(@NotNull String fourCC, @NotNull byte[] payload) { try { if (fourCC.equals(WavDirectory.CHUNK_FORMAT)) { ByteArrayReader reader = new ByteArrayReader(payload); reader.setMotorolaByteOrder(false); int wFormatTag = reader.getInt16(0); int wChannels = reader.getInt16(2); int dwSamplesPerSec = reader.getInt32(4); int dwAvgBytesPerSec = reader.getInt32(8); int wBlockAlign = reader.getInt16(12); switch (wFormatTag) { // Microsoft Pulse Code Modulation (PCM) case (0x0001): int wBitsPerSample = reader.getInt16(14); _directory.setInt(WavDirectory.TAG_BITS_PER_SAMPLE, wBitsPerSample); _directory.setString(WavDirectory.TAG_FORMAT, WavDirectory._audioEncodingMap.get(wFormatTag)); break; default: if (WavDirectory._audioEncodingMap.containsKey(wFormatTag)) { _directory.setString(WavDirectory.TAG_FORMAT, WavDirectory._audioEncodingMap.get(wFormatTag)); } else { _directory.setString(WavDirectory.TAG_FORMAT, "Unknown"); } } _directory.setInt(WavDirectory.TAG_CHANNELS, wChannels); _directory.setInt(WavDirectory.TAG_SAMPLES_PER_SEC, dwSamplesPerSec); _directory.setInt(WavDirectory.TAG_BYTES_PER_SEC, dwAvgBytesPerSec); _directory.setInt(WavDirectory.TAG_BLOCK_ALIGNMENT, wBlockAlign); } else if (fourCC.equals(WavDirectory.CHUNK_DATA)) { try { if (_directory.containsTag(WavDirectory.TAG_BYTES_PER_SEC)) { double duration = (double)payload.length / _directory.getDouble(WavDirectory.TAG_BYTES_PER_SEC); Integer hours = (int)duration / (int)(Math.pow(60, 2)); Integer minutes = ((int)duration / (int)(Math.pow(60, 1))) - (hours * 60); Integer seconds = (int)Math.round((duration / (Math.pow(60, 0))) - (minutes * 60)); String time = String.format("%1$02d:%2$02d:%3$02d", hours, minutes, seconds); _directory.setString(WavDirectory.TAG_DURATION, time); } } catch (MetadataException ex) { _directory.addError("Error calculating duration: bytes per second not found"); } }else if (WavDirectory._tagIntegerMap.containsKey(fourCC)) { _directory.setString(WavDirectory._tagIntegerMap.get(fourCC), new String(payload).substring(0, payload.length - 1)); } } catch (IOException ex) { _directory.addError(ex.getMessage()); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy