Merge master.
[dcpomatic.git] / src / lib / player.cc
index 77630f0e3c8e4e5d4f442351ae05e876d7d78f00..48c75078e12a59c2bdfabab7f5a91bbf1ea085f0 100644 (file)
 #include "sndfile_decoder.h"
 #include "sndfile_content.h"
 #include "subtitle_content.h"
+#include "subrip_decoder.h"
+#include "subrip_content.h"
 #include "playlist.h"
 #include "job.h"
 #include "image.h"
 #include "ratio.h"
 #include "log.h"
 #include "scaler.h"
+#include "render_subtitles.h"
 
 using std::list;
 using std::cout;
@@ -167,7 +170,8 @@ Player::pass ()
 
        shared_ptr<DecodedVideo> dv = dynamic_pointer_cast<DecodedVideo> (earliest_decoded);
        shared_ptr<DecodedAudio> da = dynamic_pointer_cast<DecodedAudio> (earliest_decoded);
-       shared_ptr<DecodedSubtitle> ds = dynamic_pointer_cast<DecodedSubtitle> (earliest_decoded);
+       shared_ptr<DecodedImageSubtitle> dis = dynamic_pointer_cast<DecodedImageSubtitle> (earliest_decoded);
+       shared_ptr<DecodedTextSubtitle> dts = dynamic_pointer_cast<DecodedTextSubtitle> (earliest_decoded);
 
        /* Will be set to false if we shouldn't consume the peeked DecodedThing */
        bool consume = true;
@@ -231,10 +235,14 @@ Player::pass ()
                        _statistics.audio.skip += da->data->frames();
                }
                
-       } else if (ds && _video) {
-               _in_subtitle.piece = earliest_piece;
-               _in_subtitle.subtitle = ds;
-               update_subtitle ();
+       } else if (dis && _video) {
+               _image_subtitle.piece = earliest_piece;
+               _image_subtitle.subtitle = dis;
+               update_subtitle_from_image ();
+       } else if (dts && _video) {
+               _text_subtitle.piece = earliest_piece;
+               _text_subtitle.subtitle = dts;
+               update_subtitle_from_text ();
        }
 
        if (consume) {
@@ -442,15 +450,45 @@ Player::setup_pieces ()
 
        for (ContentList::iterator i = content.begin(); i != content.end(); ++i) {
 
+               if (!(*i)->paths_valid ()) {
+                       continue;
+               }
+               
                shared_ptr<Decoder> decoder;
                optional<FrameRateChange> frc;
 
+               /* Work out a FrameRateChange for the best overlap video for this content, in case we need it below */
+               DCPTime best_overlap_t = 0;
+               shared_ptr<VideoContent> best_overlap;
+               for (ContentList::iterator j = content.begin(); j != content.end(); ++j) {
+                       shared_ptr<VideoContent> vc = dynamic_pointer_cast<VideoContent> (*j);
+                       if (!vc) {
+                               continue;
+                       }
+                       
+                       DCPTime const overlap = max (vc->position(), (*i)->position()) - min (vc->end(), (*i)->end());
+                       if (overlap > best_overlap_t) {
+                               best_overlap = vc;
+                               best_overlap_t = overlap;
+                       }
+               }
+
+               optional<FrameRateChange> best_overlap_frc;
+               if (best_overlap) {
+                       best_overlap_frc = FrameRateChange (best_overlap->video_frame_rate(), _film->video_frame_rate ());
+               } else {
+                       /* No video overlap; e.g. if the DCP is just audio */
+                       best_overlap_frc = FrameRateChange (_film->video_frame_rate(), _film->video_frame_rate ());
+               }
+
+               /* FFmpeg */
                shared_ptr<const FFmpegContent> fc = dynamic_pointer_cast<const FFmpegContent> (*i);
                if (fc) {
                        decoder.reset (new FFmpegDecoder (_film, fc, _video, _audio));
                        frc = FrameRateChange (fc->video_frame_rate(), _film->video_frame_rate());
                }
-               
+
+               /* ImageContent */
                shared_ptr<const ImageContent> ic = dynamic_pointer_cast<const ImageContent> (*i);
                if (ic) {
                        /* See if we can re-use an old ImageDecoder */
@@ -468,36 +506,18 @@ Player::setup_pieces ()
                        frc = FrameRateChange (ic->video_frame_rate(), _film->video_frame_rate());
                }
 
+               /* SndfileContent */
                shared_ptr<const SndfileContent> sc = dynamic_pointer_cast<const SndfileContent> (*i);
                if (sc) {
                        decoder.reset (new SndfileDecoder (_film, sc));
+                       frc = best_overlap_frc;
+               }
 
-                       /* Working out the frc for this content is a bit tricky: what if it overlaps
-                          two pieces of video content with different frame rates?  For now, use
-                          the one with the best overlap.
-                       */
-
-                       DCPTime best_overlap_t = 0;
-                       shared_ptr<VideoContent> best_overlap;
-                       for (ContentList::iterator j = content.begin(); j != content.end(); ++j) {
-                               shared_ptr<VideoContent> vc = dynamic_pointer_cast<VideoContent> (*j);
-                               if (!vc) {
-                                       continue;
-                               }
-
-                               DCPTime const overlap = max (vc->position(), sc->position()) - min (vc->end(), sc->end());
-                               if (overlap > best_overlap_t) {
-                                       best_overlap = vc;
-                                       best_overlap_t = overlap;
-                               }
-                       }
-
-                       if (best_overlap) {
-                               frc = FrameRateChange (best_overlap->video_frame_rate(), _film->video_frame_rate ());
-                       } else {
-                               /* No video overlap; e.g. if the DCP is just audio */
-                               frc = FrameRateChange (_film->video_frame_rate(), _film->video_frame_rate ());
-                       }
+               /* SubRipContent */
+               shared_ptr<const SubRipContent> rc = dynamic_pointer_cast<const SubRipContent> (*i);
+               if (rc) {
+                       decoder.reset (new SubRipDecoder (_film, rc));
+                       frc = best_overlap_frc;
                }
 
                ContentTime st = (*i)->trim_start() * frc->speed_up;
@@ -531,9 +551,14 @@ Player::content_changed (weak_ptr<Content> w, int property, bool frequent)
                _have_valid_pieces = false;
                Changed (frequent);
 
-       } else if (property == SubtitleContentProperty::SUBTITLE_OFFSET || property == SubtitleContentProperty::SUBTITLE_SCALE) {
+       } else if (
+               property == SubtitleContentProperty::SUBTITLE_X_OFFSET ||
+               property == SubtitleContentProperty::SUBTITLE_Y_OFFSET ||
+               property == SubtitleContentProperty::SUBTITLE_SCALE
+               ) {
 
-               update_subtitle ();
+               update_subtitle_from_image ();
+               update_subtitle_from_text ();
                Changed (frequent);
 
        } else if (
@@ -545,6 +570,7 @@ Player::content_changed (weak_ptr<Content> w, int property, bool frequent)
 
        } else if (property == ContentProperty::PATH) {
 
+               _have_valid_pieces = false;
                Changed (frequent);
        }
 }
@@ -616,14 +642,14 @@ Player::film_changed (Film::Property p)
 }
 
 void
-Player::update_subtitle ()
+Player::update_subtitle_from_image ()
 {
-       shared_ptr<Piece> piece = _in_subtitle.piece.lock ();
+       shared_ptr<Piece> piece = _image_subtitle.piece.lock ();
        if (!piece) {
                return;
        }
 
-       if (!_in_subtitle.subtitle->image) {
+       if (!_image_subtitle.subtitle->image) {
                _out_subtitle.image.reset ();
                return;
        }
@@ -631,10 +657,11 @@ Player::update_subtitle ()
        shared_ptr<SubtitleContent> sc = dynamic_pointer_cast<SubtitleContent> (piece->content);
        assert (sc);
 
-       dcpomatic::Rect<double> in_rect = _in_subtitle.subtitle->rect;
+       dcpomatic::Rect<double> in_rect = _image_subtitle.subtitle->rect;
        libdcp::Size scaled_size;
 
-       in_rect.y += sc->subtitle_offset ();
+       in_rect.x += sc->subtitle_x_offset ();
+       in_rect.y += sc->subtitle_y_offset ();
 
        /* We will scale the subtitle up to fit _video_container_size, and also by the additional subtitle_scale */
        scaled_size.width = in_rect.width * _video_container_size.width * sc->subtitle_scale ();
@@ -654,30 +681,16 @@ Player::update_subtitle ()
        
        _out_subtitle.position.x = rint (_video_container_size.width * (in_rect.x + (in_rect.width * (1 - sc->subtitle_scale ()) / 2)));
        _out_subtitle.position.y = rint (_video_container_size.height * (in_rect.y + (in_rect.height * (1 - sc->subtitle_scale ()) / 2)));
-
-       _out_subtitle.image = _in_subtitle.subtitle->image->scale (
+       
+       _out_subtitle.image = _image_subtitle.subtitle->image->scale (
                scaled_size,
                Scaler::from_id ("bicubic"),
-               PIX_FMT_RGBA,
+               _image_subtitle.subtitle->image->pixel_format (),
                true
                );
-
-<<<<<<< HEAD
-       _out_subtitle.from = _in_subtitle.subtitle->dcp_time;
-       _out_subtitle.to = _in_subtitle.subtitle->dcp_time_to;
-=======
-       /* XXX: hack */
-       Time from = _in_subtitle.from;
-       Time to = _in_subtitle.to;
-       shared_ptr<VideoContent> vc = dynamic_pointer_cast<VideoContent> (piece->content);
-       if (vc) {
-               from = rint (from * vc->video_frame_rate() / _film->video_frame_rate());
-               to = rint (to * vc->video_frame_rate() / _film->video_frame_rate());
-       }
        
-       _out_subtitle.from = from * piece->content->position ();
-       _out_subtitle.to = to + piece->content->position ();
->>>>>>> master
+       _out_subtitle.from = _image_subtitle.subtitle->dcp_time + piece->content->position ();
+       _out_subtitle.to = _image_subtitle.subtitle->dcp_time_to + piece->content->position ();
 }
 
 /** Re-emit the last frame that was emitted, using current settings for crop, ratio, scaler and subtitles.
@@ -698,13 +711,23 @@ Player::repeat_last_video ()
        return true;
 }
 
+void
+Player::update_subtitle_from_text ()
+{
+       if (_text_subtitle.subtitle->subs.empty ()) {
+               _out_subtitle.image.reset ();
+               return;
+       }
+
+       render_subtitles (_text_subtitle.subtitle->subs, _video_container_size, _out_subtitle.image, _out_subtitle.position);
+}
+
 void
 Player::set_approximate_size ()
 {
        _approximate_size = true;
 }
                              
-
 PlayerImage::PlayerImage (
        shared_ptr<const Image> in,
        Crop crop,