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

com.aspectran.demo.skylark.tts.ByteStreamAudioPlayer Maven / Gradle / Ivy

/*
 * Copyright (c) 2008-2024 The Aspectran Project
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package com.aspectran.demo.skylark.tts;

import com.aspectran.utils.annotation.jsr305.NonNull;
import com.sun.speech.freetts.audio.AudioPlayer;

import javax.sound.sampled.AudioFileFormat;
import javax.sound.sampled.AudioFormat;
import javax.sound.sampled.AudioInputStream;
import javax.sound.sampled.AudioSystem;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.OutputStream;

/**
 * Provides an implementation of AudioPlayer that sends
 * all audio data to the bit bucket. The ByteArrayAudioPlayer
 * is a helper, targeted at obtaining a byte array from the audio stream.
 */
public class ByteStreamAudioPlayer implements AudioPlayer {

    private final OutputStream output;

    private AudioFormat audioFormat;

    private Transformer transformer;

    private byte[] outputData;

    private int currIndex = 0;

    private int totalBytes = 0;

    /**
     * Constructs a ByteStreamAudioPlayer.
     */
    public ByteStreamAudioPlayer(OutputStream output) {
        this.output = output;
    }

    /**
     * Sets the audio format for this player.
     * @param format the audio format
     */
    @Override
    public void setAudioFormat(AudioFormat format) {
        this.audioFormat = format;
    }

    /**
     * Retrieves the audio format for this player.
     * @return the current audio format
     */
    @Override
    public AudioFormat getAudioFormat() {
        return audioFormat;
    }

    public void setTransformer(Transformer transformer) {
        this.transformer = transformer;
    }

    /**
     * Pauses the audio output.
     */
    @Override
    public void pause() {
    }

    /**
     * Resumes audio output.
     */
    @Override
    public synchronized void resume() {
    }

    /**
     * Cancels all queued output. Current 'write' call will return
     * false.
     */
    @Override
    public synchronized void cancel() {
    }

    /**
     * Prepares for another batch of output. Larger groups of output
     * (such as all output associated with a single FreeTTSSpeakable)
     * should be grouped between a reset/drain pair.
     */
    @Override
    public synchronized void reset() {
    }

    /**
     * Waits for all audio playback to stop, and closes this AudioPlayer.
     */
    @Override
    public synchronized void close() {
    }

    /**
     * Returns the current volume.
     * @return the current volume (between 0 and 1)
     */
    @Override
    public float getVolume() {
        return 1.0f;
    }

    /**
     * Sets the current volume.
     * @param volume the current volume (between 0 and 1)
     */
    @Override
    public void setVolume(float volume) {
    }

    /**
     * Waits for all queued audio to be played
     * @return true if the audio played to completion,
     *  false if the audio was stopped
     */
    @Override
    public boolean drain() {
        return true;
    }

    /**
     * Gets the amount of played since the last resetTime
     * Currently not supported.
     * @return the amount of audio in milliseconds
     */
    @Override
    public long getTime() {
        return -1L;
    }

    /**
     * Resets the audio clock.
     */
    @Override
    public void resetTime() {
    }

    /**
     * Starts the first sample timer.
     */
    @Override
    public void startFirstSampleTimer() {
    }

    /**
     * Shows metrics for this audio player.
     */
    @Override
    public void showMetrics() {
    }

    /**
     * Starts the output of a set of data.
     * @param size the size of data between now and the end
     */
    @Override
    public void begin(int size) {
        outputData = new byte[size];
        currIndex = 0;
    }

    /**
     * Marks the end of a set of data.
     */
    @Override
    public boolean end() {
        totalBytes += outputData.length;
        try {
            byte[] audioData = makeAudioData();
            if (transformer != null) {
                audioData = transformer.transform(audioData);
            }
            output.write(audioData);
        } catch (IOException e) {
            // ignore
        }
        return true;
    }

    /**
     * Writes the given bytes to the audio stream.
     * @param audioData array of audio data
     * @return true of the write completed successfully,
     * false if the write was cancelled.
     */
    @Override
    public boolean write(byte[] audioData) {
        return write(audioData, 0, audioData.length);
    }

    /**
     * Writes the given bytes to the audio stream.
     * @param bytes  audio data to write to the device
     * @param offset the offset into the buffer
     * @param size   the size into the buffer
     * @return true of the write completed successfully,
     * false if the write was cancelled.
     */
    @Override
    public boolean write(byte[] bytes, int offset, int size) {
        System.arraycopy(bytes, offset, outputData, currIndex, size);
        currIndex += size;
        return true;
    }

    /**
     * Provide the audio data that has been written to this AudioPlayer since
     * the last call to begin() as a byte array.
     */
    @NonNull
    private byte[] makeAudioData() throws IOException {
        AudioFormat af = getAudioFormat();
        if (af == null) {
            af = new AudioFormat(16000.0f, 16, 1, true, true);
        }
        long lengthInSamples = totalBytes / af.getFrameSize();
        AudioInputStream ais = new AudioInputStream(new ByteArrayInputStream(outputData), af, lengthInSamples);
        ByteArrayOutputStream out = new ByteArrayOutputStream();
        AudioSystem.write(ais, AudioFileFormat.Type.WAVE, out);
        return out.toByteArray();
    }

    public int getTotalBytes() {
        return totalBytes;
    }

    /**
     * Returns the name of this AudioPlayer.
     * @return the name of the audio player
     */
    @Override
    public String toString() {
        return "ByteStreamAudioPlayer";
    }

    /**
     * A transformer that encodes bytes into different types of data.
     */
    public interface Transformer {

        byte[] transform(byte[] bytes);

    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy