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

net.sourceforge.javaflacencoder.FLAC_FileEncoder Maven / Gradle / Ivy

Go to download

A port of the Free Lossless Audio Codec (FLAC) decoder to Java and a FLAC encoder implemented in Java.

There is a newer version: 1.4.1
Show newest version
/*
 * Copyright (C) 2010  Preston Lacey http://javaflacencoder.sourceforge.net/
 * All Rights Reserved.
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
 */

package net.sourceforge.javaflacencoder;
import java.io.File;
import java.io.IOException;

import javax.sound.sampled.AudioFormat;
import javax.sound.sampled.AudioInputStream;
import javax.sound.sampled.AudioSystem;
import javax.sound.sampled.UnsupportedAudioFileException;
/**
 * FLAC_FileEncoder is a class to encode an input wav File to an output Flac
 * file. It allows the EncodingConfiguration to be set only once, prior to
 * encoding the entire File.
 * 
 * @author Preston Lacey
 */
public class FLAC_FileEncoder {
  /** Maximum number of bytes to read from file at once */
  private static final int MAX_READ = 16384;

  /** Status enum for encode result */
  public enum Status {
    /** Unknown State. */
    UNKNOWN,
    /** Everything went well */
    FULL_ENCODE,

    /** Something unspecified went wrong...*/
    GENERAL_ERROR,

    /** internal error is something that went haywire that was discovered
     * due to internal sanity checks. A problem in API. */
    INTERNAL_ERROR,

    /** File given was not able to be read */
    UNSUPPORTED_FILE,

    /** Generic file IO Error */
    FILE_IO_ERROR,

    /** Sample size unsupported */
    UNSUPPORTED_SAMPLE_SIZE,

    /** Error with output file */
    OUTPUT_FILE_ERROR,
    /** No errors found. */
    OK
  }
  FLACEncoder flac = null;
  StreamConfiguration sc = null;
  EncodingConfiguration ec = null;
  File outFile = null;
  int lastTotalSamples = 0;
  boolean useThreads;

  /**
   * Constructor creates a FLAC_FileEncoder object with default
   * StreamConfiguration configuration and default EncodingConfiguration.
   * Thread use defaults to true.
   */
  public FLAC_FileEncoder() {
    flac = new FLACEncoder();
    sc = new StreamConfiguration();
    ec = new EncodingConfiguration();
    useThreads = true;
  }

  /**
   * Specify whether to use multiple threads or not.
   * @param val true to use threads, false otherwise.
   */
  public void useThreads(boolean val) {
    useThreads = val;
  }

  private void adjustConfigurations(AudioFormat format) {
    int sampleRate = (int)format.getSampleRate();
    int sampleSize = (int)format.getSampleSizeInBits();
    int channels = (int)format.getChannels();
    sc.setSampleRate(sampleRate);
    sc.setBitsPerSample(sampleSize);
    sc.setChannelCount(channels);
  }

  /**
   * Set the stream configuration for this encoder to use. Note that the audio
   * characteristics(number of channels, sample rate, and sample size), will
   * be set to match the input file at encode time, so needn't be set in the
   * given StreamConfiguration object.
   *
   * @param config StreamConfiguration to use for encoding.
   */
  public void setStreamConfig(StreamConfiguration config) {sc = config; }

  /**
   * Set the EncodingConfiguration to use for encoding.
   * @param config EncodingConfiguration to use.
   */
  public void setEncodingConfig(EncodingConfiguration config){ec = config;}

  private Status openStream() {
    Status status = Status.OK;
    boolean result = flac.setStreamConfiguration(sc);
    result = result & flac.setEncodingConfiguration(ec);
    if( !result)
      status = Status.INTERNAL_ERROR;
    else {
      FLACFileOutputStream fout = null;
      try{
        fout = new FLACFileOutputStream(outFile.getPath());
      } catch(IOException e) {
        status = Status.OUTPUT_FILE_ERROR;
      }
      if( status == Status.OK) {
        flac.setOutputStream(fout);
        try {
          flac.openFLACStream();
        }catch(IOException e) {
          status = Status.INTERNAL_ERROR;
        }
      }
      else
        status = Status.OUTPUT_FILE_ERROR;
    }
    return status;
  }

  /**
   * Encode the given input wav file to an output file.
   *
   * @param inputFile Input wav file to encode.
   * @param outputFile Output file to write FLAC stream to. If file exists, it
   * will be overwritten without prompting.
   *
   * @return Status flag for encode
   */
  public Status encode(File inputFile, File outputFile) {
    Status status = Status.FULL_ENCODE;
    this.outFile = outputFile;
    //take file and initial configuration.
    //open file
    AudioInputStream sin = null;
    AudioFormat format = null;
    try {
      sin = AudioSystem.getAudioInputStream(inputFile);
    }catch(IOException e) {
      status = Status.FILE_IO_ERROR;
    }catch (UnsupportedAudioFileException e) {
      status = Status.UNSUPPORTED_FILE;
    }finally {
      if(status != Status.FULL_ENCODE)
      {
        try {
          sin.close();
        }catch (IOException e) {
          //Nothing to do, close failed
        }
        return status;
      }
    }
    try {
      format = sin.getFormat();
      //sanitize and optimize configurations
      adjustConfigurations(format);
      //open stream
      openStream();
      AudioStreamEncoder.encodeAudioInputStream(sin, MAX_READ, flac, useThreads);
    }
    catch(IOException e) {
      status = Status.FILE_IO_ERROR;
    }
    catch(IllegalArgumentException e) {
      status = Status.GENERAL_ERROR;
      String message = e.getMessage();
      if(message.equals(Status.UNSUPPORTED_SAMPLE_SIZE.name()))
        status = Status.UNSUPPORTED_SAMPLE_SIZE;
      else
        throw e;
    }
    finally {
      try {
        sin.close();
      }catch (IOException e) {
        //Nothing to do, close failed
      }
    }
    
    return status;
  }

  /**
   * Get the total number of samples encoded in last encode.  This is here
   * primarily for use as a sanity check during debugging.
   *
   * @return Total number of samples encoded in last encode attempt.
   */
  public int getLastTotalSamplesEncoded() {
    return this.lastTotalSamples;
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy