int main()

in vireo/tools/stitch/main.cpp [44:149]


int main(int argc, const char* argv[]) {
  // Usage
  auto print_usage = [](const string name) {
    const int opt_len = 30;
    cout << "Usage: " << name << " [options] infiles outfile" << endl;
    cout << "\nOptions:" << endl;
    cout << std::left << std::setw(opt_len) << "--disable_audio"            << "disable audio track (default: no)" << endl;
    cout << std::left << std::setw(opt_len) << "--help"                     << "show usage" << endl;
    cout << std::left << std::setw(opt_len) << "--version"                  << "show version" << endl;
  };

  const string name = common::Path::Filename(argv[0]);
  if (argc < 2) {
    print_usage(name);
    return 1;
  }

  // Parse arguments
  bool disable_audio = false;
  int last_arg = 1;
  for (int i = 1; i < argc; ++i) {
    if (strcmp(argv[i], "--disable_audio") == 0) {
      disable_audio = true;
      last_arg = i + 1;
    } else if (strcmp(argv[i], "--version") == 0) {
      cout << name << " version " << STITCH_VERSION << " (based on vireo " << VIREO_VERSION << ")" << endl;
      return 0;
    } else if (strcmp(argv[i], "--help") == 0) {
      print_usage(name);
      return 0;
    }
  }
  if (last_arg + 1 >= argc) {
    cerr << "Need to specify infiles and outfile" << endl;
    return 1;
  }

  __try {
    // Create decoders
    vector<string> filenames;
    for (int i = last_arg; i < argc - 1; ++i) {
      filenames.push_back(argv[i]);
    }

    vector<functional::Audio<decode::Sample>> audios;
    vector<functional::Video<decode::Sample>> videos;
    vector<vector<common::EditBox>> edit_boxes_per_track;

    // Ensure videos are compatible for stitching
    auto demuxer_0 = demux::Movie(filenames[0]);
    for (auto filename: filenames) {
      auto demuxer = demux::Movie(filename);
      if (!demuxer.video_track.count()) {
        cerr << "Could not find video track: " << filename << endl;
        return 1;
      }
      if (!disable_audio && demuxer_0.audio_track.settings().timescale != demuxer.audio_track.settings().timescale) {
        cerr << "Audio timescale does not match: " << filenames[0] << " and " << filename << endl;
        cerr << "Use --disable_audio to disable stitching audio tracks" << endl;
        return 1;
      }
      if (!disable_audio && demuxer_0.audio_track.settings().sample_rate != demuxer.audio_track.settings().sample_rate) {
        cerr << "Audio sample rate does not match: " << filenames[0] << " and " << filename << endl;
        cerr << "Use --disable_audio to disable stitching audio tracks" << endl;
        return 1;
      }
      if (demuxer_0.video_track.settings().width != demuxer.video_track.settings().width ||
          demuxer_0.video_track.settings().height != demuxer.video_track.settings().height) {
        cerr << "Dimensions do not match: " << filenames[0] << " (" << demuxer_0.video_track.settings().width << ", " << demuxer_0.video_track.settings().height <<  ") and ";
        cerr << filename <<  " (" << demuxer.video_track.settings().width << ", " << demuxer.video_track.settings().height <<  ")" << endl;
        cerr << "Transcode the video to allow stitching" << endl;
        return 1;
      }
      if (demuxer_0.video_track.settings().sps_pps.pps != demuxer.video_track.settings().sps_pps.pps ||
          demuxer_0.video_track.settings().sps_pps.sps != demuxer.video_track.settings().sps_pps.sps) {
        cerr << "Incompatible SPS or PPS" << endl;
        return 1;
      }
      videos.push_back((functional::Video<decode::Sample>)demuxer.video_track);
      vector<common::EditBox> edit_boxes = demuxer.video_track.edit_boxes();
      if (!disable_audio) {
        audios.push_back((functional::Audio<decode::Sample>)demuxer.audio_track);
        edit_boxes.insert(edit_boxes.end(), demuxer.audio_track.edit_boxes().begin(), demuxer.audio_track.edit_boxes().end());
      }
      edit_boxes_per_track.push_back(edit_boxes);
    }

    // Stitch
    auto stitched = transform::Stitch(audios, videos, edit_boxes_per_track);
    vector<common::EditBox> edit_boxes = stitched.audio_track.edit_boxes();
    edit_boxes.insert(edit_boxes.end(), stitched.video_track.edit_boxes().begin(), stitched.video_track.edit_boxes().end());
    mux::MP4 muxer(functional::Audio<encode::Sample>(stitched.audio_track, encode::Sample::Convert),
                   functional::Video<encode::Sample>(stitched.video_track, encode::Sample::Convert),
                   edit_boxes);
    util::save(vireo::common::Path::MakeAbsolute(argv[argc - 1]), muxer());
  } __catch (std::exception& e) {
#if __EXCEPTIONS
    cerr << "Error stitching movie: " << e.what() << endl;
#else
    cerr << "Error stitching movie" << endl;
#endif
    return 1;
  }

  return 0;
}