001: /*
002: * Copyright (c) 2001 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: * @Author: Silvere Martin-Michiellot
032: *
033: */
034:
035: // This code is repackaged after the Sun Microsystem demo available as a JMF Solution
036: // Site http://www.java.sun.com/
037: // Email
038: package com.db.media;
039:
040: import java.io.*;
041: import java.util.*;
042: import java.awt.Dimension;
043:
044: import javax.media.*;
045: import javax.media.control.*;
046: import javax.media.protocol.*;
047: import javax.media.protocol.DataSource;
048: import javax.media.datasink.*;
049: import javax.media.format.VideoFormat;
050:
051: /**
052: * This program takes a list of JPEG image files and convert them into
053: * a QuickTime movie.
054: */
055: public class JpegImagesToMovie implements ControllerListener,
056: DataSinkListener {
057:
058: Object waitSync = new Object();
059: boolean stateTransitionOK = true;
060:
061: Object waitFileSync = new Object();
062: boolean fileDone = false;
063: boolean fileSuccess = true;
064:
065: public boolean doIt(int width, int height, int frameRate,
066: Vector inFiles, MediaLocator outML) {
067:
068: ImageDataSource ids = new ImageDataSource(width, height,
069: frameRate, inFiles);
070:
071: Processor p;
072:
073: try {
074: System.err
075: .println("- create processor for the image datasource ...");
076: p = Manager.createProcessor(ids);
077: } catch (Exception e) {
078: System.err
079: .println("Yikes! Cannot create a processor from the data source.");
080: return false;
081: }
082:
083: p.addControllerListener(this );
084:
085: // Put the Processor into configured state so we can set
086: // some processing options on the processor.
087: p.configure();
088: if (!waitForState(p, p.Configured)) {
089: System.err.println("Failed to configure the processor.");
090: return false;
091: }
092:
093: // Set the output content descriptor to QuickTime.
094: p.setContentDescriptor(new ContentDescriptor(
095: FileTypeDescriptor.QUICKTIME));
096:
097: // Query for the processor for supported formats.
098: // Then set it on the processor.
099: TrackControl tcs[] = p.getTrackControls();
100: Format f[] = tcs[0].getSupportedFormats();
101: if (f == null || f.length <= 0) {
102: System.err
103: .println("The mux does not support the input format: "
104: + tcs[0].getFormat());
105: return false;
106: }
107:
108: tcs[0].setFormat(f[0]);
109:
110: System.err.println("Setting the track format to: " + f[0]);
111:
112: // We are done with programming the processor. Let's just
113: // realize it.
114: p.realize();
115: if (!waitForState(p, p.Realized)) {
116: System.err.println("Failed to realize the processor.");
117: return false;
118: }
119:
120: // Now, we'll need to create a DataSink.
121: DataSink dsink;
122: if ((dsink = createDataSink(p, outML)) == null) {
123: System.err
124: .println("Failed to create a DataSink for the given output MediaLocator: "
125: + outML);
126: return false;
127: }
128:
129: dsink.addDataSinkListener(this );
130: fileDone = false;
131:
132: System.err.println("start processing...");
133:
134: // OK, we can now start the actual transcoding.
135: try {
136: p.start();
137: dsink.start();
138: } catch (IOException e) {
139: System.err.println("IO error during processing");
140: return false;
141: }
142:
143: // Wait for EndOfStream event.
144: waitForFileDone();
145:
146: // Cleanup.
147: try {
148: dsink.close();
149: } catch (Exception e) {
150: }
151: p.removeControllerListener(this );
152:
153: System.err.println("...done processing.");
154:
155: return true;
156:
157: }
158:
159: /**
160: * Create the DataSink.
161: */
162: DataSink createDataSink(Processor p, MediaLocator outML) {
163:
164: DataSource ds;
165:
166: if ((ds = p.getDataOutput()) == null) {
167: System.err
168: .println("Something is really wrong: the processor does not have an output DataSource");
169: return null;
170: }
171:
172: DataSink dsink;
173:
174: try {
175: System.err.println("- create DataSink for: " + outML);
176: dsink = Manager.createDataSink(ds, outML);
177: dsink.open();
178: } catch (Exception e) {
179: System.err.println("Cannot create the DataSink: " + e);
180: return null;
181: }
182:
183: return dsink;
184:
185: }
186:
187: /**
188: * Block until the processor has transitioned to the given state.
189: * Return false if the transition failed.
190: */
191: boolean waitForState(Processor p, int state) {
192:
193: synchronized (waitSync) {
194: try {
195: while (p.getState() < state && stateTransitionOK)
196: waitSync.wait();
197: } catch (Exception e) {
198: }
199: }
200:
201: return stateTransitionOK;
202:
203: }
204:
205: /**
206: * Controller Listener.
207: */
208: public void controllerUpdate(ControllerEvent evt) {
209:
210: if (evt instanceof ConfigureCompleteEvent
211: || evt instanceof RealizeCompleteEvent
212: || evt instanceof PrefetchCompleteEvent) {
213: synchronized (waitSync) {
214: stateTransitionOK = true;
215: waitSync.notifyAll();
216: }
217: } else if (evt instanceof ResourceUnavailableEvent) {
218: synchronized (waitSync) {
219: stateTransitionOK = false;
220: waitSync.notifyAll();
221: }
222: } else if (evt instanceof EndOfMediaEvent) {
223: evt.getSourceController().stop();
224: evt.getSourceController().close();
225: }
226:
227: }
228:
229: /**
230: * Block until file writing is done.
231: */
232: boolean waitForFileDone() {
233:
234: synchronized (waitFileSync) {
235: try {
236: while (!fileDone)
237: waitFileSync.wait();
238: } catch (Exception e) {
239: }
240: }
241:
242: return fileSuccess;
243:
244: }
245:
246: /**
247: * Event handler for the file writer.
248: */
249: public void dataSinkUpdate(DataSinkEvent evt) {
250:
251: if (evt instanceof EndOfStreamEvent) {
252: synchronized (waitFileSync) {
253: fileDone = true;
254: waitFileSync.notifyAll();
255: }
256: } else if (evt instanceof DataSinkErrorEvent) {
257: synchronized (waitFileSync) {
258: fileDone = true;
259: fileSuccess = false;
260: waitFileSync.notifyAll();
261: }
262: }
263:
264: }
265:
266: public static void main(String args[]) {
267:
268: if (args.length == 0)
269: prUsage();
270:
271: // Parse the arguments.
272: int i = 0;
273: int width = -1, height = -1, frameRate = 1;
274: Vector inputFiles = new Vector();
275: String outputURL = null;
276:
277: while (i < args.length) {
278:
279: if (args[i].equals("-w")) {
280: i++;
281: if (i >= args.length)
282: prUsage();
283: width = new Integer(args[i]).intValue();
284: } else if (args[i].equals("-h")) {
285: i++;
286: if (i >= args.length)
287: prUsage();
288: height = new Integer(args[i]).intValue();
289: } else if (args[i].equals("-f")) {
290: i++;
291: if (i >= args.length)
292: prUsage();
293: frameRate = new Integer(args[i]).intValue();
294: } else if (args[i].equals("-o")) {
295: i++;
296: if (i >= args.length)
297: prUsage();
298: outputURL = args[i];
299: } else {
300: inputFiles.addElement(args[i]);
301: }
302: i++;
303: }
304:
305: if (outputURL == null || inputFiles.size() == 0)
306: prUsage();
307:
308: // Check for output file extension.
309: if (!outputURL.endsWith(".mov") && !outputURL.endsWith(".MOV")) {
310: System.err
311: .println("The output file extension should end with a .mov extension");
312: prUsage();
313: }
314:
315: if (width < 0 || height < 0) {
316: System.err
317: .println("Please specify the correct image size.");
318: prUsage();
319: }
320:
321: // Check the frame rate.
322: if (frameRate < 1)
323: frameRate = 1;
324:
325: // Generate the output media locators.
326: MediaLocator oml;
327:
328: if ((oml = createMediaLocator(outputURL)) == null) {
329: System.err.println("Cannot build media locator from: "
330: + outputURL);
331: System.exit(0);
332: }
333:
334: JpegImagesToMovie imageToMovie = new JpegImagesToMovie();
335: imageToMovie.doIt(width, height, frameRate, inputFiles, oml);
336:
337: System.exit(0);
338:
339: }
340:
341: static void prUsage() {
342:
343: System.err
344: .println("Usage: java JpegImagesToMovie -w <width> -h <height> -f <frame rate> -o <output URL> <input JPEG file 1> <input JPEG file 2> ...");
345: System.exit(-1);
346:
347: }
348:
349: /**
350: * Create a media locator from the given string.
351: */
352: static MediaLocator createMediaLocator(String url) {
353:
354: MediaLocator ml;
355:
356: if (url.indexOf(":") > 0
357: && (ml = new MediaLocator(url)) != null)
358: return ml;
359:
360: if (url.startsWith(File.separator)) {
361: if ((ml = new MediaLocator("file:" + url)) != null)
362: return ml;
363: } else {
364: String file = "file:" + System.getProperty("user.dir")
365: + File.separator + url;
366: if ((ml = new MediaLocator(file)) != null)
367: return ml;
368: }
369:
370: return null;
371:
372: }
373:
374: ///////////////////////////////////////////////
375: //
376: // Inner classes.
377: ///////////////////////////////////////////////
378:
379: /**
380: * A DataSource to read from a list of JPEG image files and
381: * turn that into a stream of JMF buffers.
382: * The DataSource is not seekable or positionable.
383: */
384: class ImageDataSource extends PullBufferDataSource {
385:
386: ImageSourceStream streams[];
387:
388: ImageDataSource(int width, int height, int frameRate,
389: Vector images) {
390:
391: streams = new ImageSourceStream[1];
392: streams[0] = new ImageSourceStream(width, height,
393: frameRate, images);
394:
395: }
396:
397: public void setLocator(MediaLocator source) {
398: }
399:
400: public MediaLocator getLocator() {
401:
402: return null;
403:
404: }
405:
406: /**
407: * Content type is of RAW since we are sending buffers of video
408: * frames without a container format.
409: */
410: public String getContentType() {
411:
412: return ContentDescriptor.RAW;
413:
414: }
415:
416: public void connect() {
417: }
418:
419: public void disconnect() {
420: }
421:
422: public void start() {
423: }
424:
425: public void stop() {
426: }
427:
428: /**
429: * Return the ImageSourceStreams.
430: */
431: public PullBufferStream[] getStreams() {
432:
433: return streams;
434:
435: }
436:
437: /**
438: * We could have derived the duration from the number of
439: * frames and frame rate. But for the purpose of this program,
440: * it's not necessary.
441: */
442: public Time getDuration() {
443:
444: return DURATION_UNKNOWN;
445:
446: }
447:
448: public Object[] getControls() {
449:
450: return new Object[0];
451:
452: }
453:
454: public Object getControl(String type) {
455:
456: return null;
457:
458: }
459:
460: }
461:
462: /**
463: * The source stream to go along with ImageDataSource.
464: */
465: class ImageSourceStream implements PullBufferStream {
466:
467: Vector images;
468: int width, height;
469: VideoFormat format;
470:
471: int nextImage = 0; // index of the next image to be read.
472: boolean ended = false;
473:
474: public ImageSourceStream(int width, int height, int frameRate,
475: Vector images) {
476: this .width = width;
477: this .height = height;
478: this .images = images;
479:
480: format = new VideoFormat(VideoFormat.JPEG, new Dimension(
481: width, height), Format.NOT_SPECIFIED,
482: Format.byteArray, (float) frameRate);
483:
484: }
485:
486: /**
487: * We should never need to block assuming data are read from files.
488: */
489: public boolean willReadBlock() {
490:
491: return false;
492:
493: }
494:
495: /**
496: * This is called from the Processor to read a frame worth
497: * of video data.
498: */
499: public void read(Buffer buf) throws IOException {
500:
501: // Check if we've finished all the frames.
502: if (nextImage >= images.size()) {
503: // We are done. Set EndOfMedia.
504: System.err.println("Done reading all images.");
505: buf.setEOM(true);
506: buf.setOffset(0);
507: buf.setLength(0);
508: ended = true;
509: return;
510: }
511:
512: String imageFile = (String) images.elementAt(nextImage);
513: nextImage++;
514:
515: System.err.println(" - reading image file: " + imageFile);
516:
517: // Open a random access file for the next image.
518: RandomAccessFile raFile;
519: raFile = new RandomAccessFile(imageFile, "r");
520:
521: byte data[] = null;
522:
523: // Check the input buffer type & size.
524:
525: if (buf.getData() instanceof byte[])
526: data = (byte[]) buf.getData();
527:
528: // Check to see the given buffer is big enough for the frame.
529: if (data == null || data.length < raFile.length()) {
530: data = new byte[(int) raFile.length()];
531: buf.setData(data);
532: }
533:
534: // Read the entire JPEG image from the file.
535: raFile.readFully(data, 0, (int) raFile.length());
536:
537: System.err.println(" read " + raFile.length()
538: + " bytes.");
539:
540: buf.setOffset(0);
541: buf.setLength((int) raFile.length());
542: buf.setFormat(format);
543: buf.setFlags(buf.getFlags() | buf.FLAG_KEY_FRAME);
544:
545: // Close the random access file.
546: raFile.close();
547:
548: }
549:
550: /**
551: * Return the format of each video frame. That will be JPEG.
552: */
553: public Format getFormat() {
554:
555: return format;
556:
557: }
558:
559: public ContentDescriptor getContentDescriptor() {
560:
561: return new ContentDescriptor(ContentDescriptor.RAW);
562:
563: }
564:
565: public long getContentLength() {
566:
567: return 0;
568:
569: }
570:
571: public boolean endOfStream() {
572:
573: return ended;
574:
575: }
576:
577: public Object[] getControls() {
578:
579: return new Object[0];
580:
581: }
582:
583: public Object getControl(String type) {
584:
585: return null;
586:
587: }
588:
589: }
590:
591: }
|