001: /*
002: * Copyright (c) 2000 Silvere Martin-Michiellot All Rights Reserved.
003: *
004: * Silvere Martin-Michiellot grants you ("Licensee") a non-exclusive,
005: * royalty free, license to use, modify and redistribute this
006: * software in source and binary code form,
007: * provided that i) this copyright notice and license appear on all copies of
008: * the software; and ii) Licensee does not utilize the software in a manner
009: * which is disparaging to Silvere Martin-Michiellot.
010: *
011: * This software is provided "AS IS," without a warranty of any kind. ALL
012: * EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES, INCLUDING ANY
013: * IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE OR
014: * NON-INFRINGEMENT, ARE HEREBY EXCLUDED. Silvere Martin-Michiellot
015: * AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES
016: * SUFFERED BY LICENSEE AS A RESULT OF USING, MODIFYING
017: * OR DISTRIBUTING THE SOFTWARE OR ITS DERIVATIVES. IN NO EVENT WILL
018: * Silvere Martin-Michiellot OR ITS LICENSORS BE LIABLE
019: * FOR ANY LOST REVENUE, PROFIT OR DATA, OR FOR DIRECT,
020: * INDIRECT, SPECIAL, CONSEQUENTIAL, INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER
021: * CAUSED AND REGARDLESS OF THE THEORY OF LIABILITY, ARISING OUT OF THE USE OF
022: * OR INABILITY TO USE SOFTWARE, EVEN IF Silvere Martin-Michiellot HAS BEEN
023: * ADVISED OF THE POSSIBILITY OF SUCH DAMAGES.
024: *
025: * This software is not designed or intended for use in on-line control of
026: * aircraft, air traffic, aircraft navigation or aircraft communications; or in
027: * the design, construction, operation or maintenance of any nuclear
028: * facility. Licensee represents and warrants that it will not use or
029: * redistribute the Software for such purposes.
030: *
031: */
032:
033: package com.db.media.audio;
034:
035: import java.io.File;
036: import javax.sound.midi.*;
037: import javax.sound.sampled.*;
038:
039: public class AudioSaver {
040:
041: private static final int QUALITY_NONE = 0;
042: private static final int QUALITY_PHONE = 1;
043: private static final int QUALITY_RADIO = 2;
044: private static final int QUALITY_CD = 3;
045: private static final int DEFAULT_QUALITY = QUALITY_CD;
046:
047: /*
048: * This set of types is used if AudioSystem.getAudioFileTypes()
049: * returns an array of length 0. This is necessary because the
050: * Sun jdk1.3.0 does so (Yes, it is a bug).
051: */
052: private static final AudioFileFormat.Type[] DEFAULT_TYPES = {
053: AudioFileFormat.Type.WAVE, AudioFileFormat.Type.AU,
054: AudioFileFormat.Type.AIFF, AudioFileFormat.Type.AIFC,
055: AudioFileFormat.Type.SND, };
056:
057: public AudioSaver(int quality, File outputFile) {
058:
059: AudioFormat format;
060: AudioFileFormat.Type targetType;
061: TargetDataLine line;
062: String strMixerName;
063: String strExtension;
064:
065: switch (quality) {
066: case QUALITY_PHONE:
067: //8 kHz, 8 bit, mono
068: format = new AudioFormat(AudioFormat.Encoding.PCM_SIGNED,
069: 8000.0F, 8, 1, 1, 8000.0F, true);
070: break;
071: case QUALITY_RADIO:
072: //22.05 kHz, 8 bit, mono
073: format = new AudioFormat(AudioFormat.Encoding.PCM_SIGNED,
074: 22050.0F, 8, 1, 1, 22050.0F, true);
075: break;
076: case QUALITY_CD:
077: default:
078: // 44.1 kHz, 16 bit, stereo
079: format = new AudioFormat(AudioFormat.Encoding.PCM_SIGNED,
080: 44100.0F, 16, 2, 4, 44100.0F, true);
081: break;
082: }
083:
084: strMixerName = "DB audio mixer";
085: strExtension = "au";
086:
087: targetType = AudioFileFormat.Type.AU;
088:
089: if (strExtension != null) {
090: targetType = findTargetType(strExtension);
091: if (targetType == null) {
092: targetType = AudioFileFormat.Type.AU;
093: }
094: }
095:
096: line = getTargetDataLine(strMixerName, format,
097: AudioSystem.NOT_SPECIFIED);
098: if (line != null) {
099: DirectRecordingStream recorder = null;
100: recorder = new DirectRecordingStream(line, targetType,
101: outputFile);
102: recorder.start();
103: recorder.stop();
104: }
105:
106: }
107:
108: private static SourceDataLine getSourceDataLine(
109: String strMixerName, AudioFormat audioFormat,
110: int nBufferSize) {
111:
112: SourceDataLine line = null;
113: DataLine.Info info = new DataLine.Info(SourceDataLine.class,
114: audioFormat, nBufferSize);
115: try {
116: if (strMixerName != null) {
117: Mixer.Info mixerInfo = getMixerInfo(strMixerName);
118: Mixer mixer = AudioSystem.getMixer(mixerInfo);
119: line = (SourceDataLine) mixer.getLine(info);
120: } else {
121: line = (SourceDataLine) AudioSystem.getLine(info);
122: }
123:
124: line.open(audioFormat, nBufferSize);
125: } catch (Exception e) {
126:
127: }
128:
129: return line;
130:
131: }
132:
133: private static Mixer.Info getMixerInfo(String strMixerName) {
134:
135: Mixer.Info[] aInfos = AudioSystem.getMixerInfo();
136: for (int i = 0; i < aInfos.length; i++) {
137: if (aInfos[i].getName().equals(strMixerName)) {
138: return aInfos[i];
139: }
140: }
141:
142: return null;
143:
144: }
145:
146: /** Trying to get an audio file type for the passed extension.
147: * This works by examining all available file types. For each
148: * type, if the extension this type promisses to handle matches
149: * the extension we are trying to find a type for, this type is
150: * returned.
151: * If no appropriate type is found, null is returned.
152: */
153: private static AudioFileFormat.Type findTargetType(
154: String strExtension) {
155:
156: AudioFileFormat.Type[] aTypes = AudioSystem.getAudioFileTypes();
157: /*
158: * Workaround for a bug in the Sun jdk1.3.0.
159: */
160: if (aTypes.length == 0) {
161: aTypes = DEFAULT_TYPES;
162: }
163: for (int i = 0; i < aTypes.length; i++) {
164: if (aTypes[i].getExtension().equals(strExtension)) {
165: return aTypes[i];
166: }
167: }
168:
169: return null;
170:
171: }
172:
173: private static TargetDataLine getTargetDataLine(
174: String strMixerName, AudioFormat audioFormat,
175: int nBufferSize) {
176:
177: TargetDataLine line = null;
178: DataLine.Info info = new DataLine.Info(TargetDataLine.class,
179: audioFormat, nBufferSize);
180:
181: try {
182: if (strMixerName != null) {
183: Mixer.Info mixerInfo = getMixerInfo(strMixerName);
184: Mixer mixer = AudioSystem.getMixer(mixerInfo);
185: line = (TargetDataLine) mixer.getLine(info);
186: } else {
187: line = (TargetDataLine) AudioSystem.getLine(info);
188: }
189:
190: line.open(audioFormat, nBufferSize);
191:
192: } catch (Exception e) {
193: }
194:
195: return line;
196:
197: }
198:
199: /*
200: * This method tries to return a MidiDevice.Info whose name
201: * matches the passed name. If no matching MidiDevice.Info is
202: * found, null is returned.
203: * If forOutput is true, then only output devices are searched,
204: * otherwise only input devices.
205: */
206: private static MidiDevice.Info getMidiDeviceInfo(
207: String strDeviceName, boolean forOutput) {
208:
209: MidiDevice.Info[] aInfos = MidiSystem.getMidiDeviceInfo();
210: for (int i = 0; i < aInfos.length; i++) {
211: if (aInfos[i].getName().equals(strDeviceName)) {
212: try {
213: MidiDevice device = MidiSystem
214: .getMidiDevice(aInfos[i]);
215: boolean bAllowsInput = (device.getMaxTransmitters() != 0);
216: boolean bAllowsOutput = (device.getMaxReceivers() != 0);
217: if ((bAllowsOutput && forOutput)
218: || (bAllowsInput && !forOutput)) {
219: return aInfos[i];
220: }
221: } catch (MidiUnavailableException mue) {
222: }
223: }
224: }
225:
226: return null;
227:
228: }
229:
230: }
|