7#include <packager/media/formats/webvtt/webvtt_to_mp4_handler.h>
12#include <absl/log/check.h>
14#include <packager/macros/logging.h>
15#include <packager/macros/status.h>
16#include <packager/media/base/buffer_writer.h>
17#include <packager/media/base/timestamp_util.h>
18#include <packager/media/formats/mp4/box_buffer.h>
19#include <packager/media/formats/mp4/box_definitions.h>
20#include <packager/media/formats/webvtt/webvtt_utils.h>
27enum class DisplayActionType { ADD, REMOVE };
30 DisplayActionType type;
31 const TextSample* sample;
34std::multimap<int64_t, DisplayAction> CreateActionList(
35 int64_t segment_start,
37 const std::list<std::shared_ptr<const TextSample>>& samples) {
38 std::multimap<int64_t, DisplayAction> actions;
40 for (
const auto& sample : samples) {
46 DCHECK(PtsIsBefore(sample->start_time(), segment_end))
47 <<
"Sample start " << sample->start_time()
48 <<
" should be before segment end " << segment_end;
50 {sample->start_time(), {DisplayActionType::ADD, sample.get()}});
54 if (PtsIsBefore(sample->EndTime(), segment_end)) {
56 {sample->EndTime(), {DisplayActionType::REMOVE, sample.get()}});
63void WriteSample(
const TextSample& sample, BufferWriter* out) {
66 if (sample.id().length()) {
67 box.cue_id.cue_id = sample.id();
69 box.cue_settings.settings = WebVttSettingsToString(sample.settings());
70 box.cue_payload.cue_text = WebVttFragmentToString(sample.body());
80void WriteSamples(
const std::list<const TextSample*>& samples,
81 BufferWriter* writer) {
82 DCHECK_GE(samples.size(), 0u);
84 for (
const auto& sample : samples) {
85 WriteSample(*sample, writer);
89void WriteEmptySample(BufferWriter* writer) {
90 mp4::VTTEmptyCueBox box;
94std::shared_ptr<MediaSample> CreateMediaSample(
const BufferWriter& buffer,
97 DCHECK_GE(start_time, 0);
98 DCHECK_GT(end_time, start_time);
100 const bool kIsKeyFrame =
true;
102 std::shared_ptr<MediaSample> sample =
104 sample->set_pts(start_time);
105 sample->set_dts(start_time);
106 sample->set_duration(end_time - start_time);
112Status WebVttToMp4Handler::InitializeInternal() {
116Status WebVttToMp4Handler::Process(std::unique_ptr<StreamData> stream_data) {
117 switch (stream_data->stream_data_type) {
118 case StreamDataType::kStreamInfo:
119 return OnStreamInfo(std::move(stream_data));
120 case StreamDataType::kCueEvent:
121 return OnCueEvent(std::move(stream_data));
122 case StreamDataType::kSegmentInfo:
123 return OnSegmentInfo(std::move(stream_data));
124 case StreamDataType::kTextSample:
125 return OnTextSample(std::move(stream_data));
127 return Status(error::INTERNAL_ERROR,
128 "Invalid stream data type (" +
129 StreamDataTypeToString(stream_data->stream_data_type) +
130 ") for this WebVttToMp4 handler");
134Status WebVttToMp4Handler::OnStreamInfo(
135 std::unique_ptr<StreamData> stream_data) {
137 DCHECK(stream_data->stream_info);
139 auto clone = stream_data->stream_info->Clone();
140 clone->set_codec(kCodecWebVtt);
141 clone->set_codec_string(
"wvtt");
143 if (clone->stream_type() != kStreamText) {
144 return Status(error::MUXER_FAILURE,
"Incorrect stream type");
148 StreamData::FromStreamInfo(stream_data->stream_index, std::move(clone)));
151Status WebVttToMp4Handler::OnCueEvent(std::unique_ptr<StreamData> stream_data) {
153 DCHECK(stream_data->cue_event);
155 if (current_segment_.size()) {
156 return Status(error::INTERNAL_ERROR,
157 "Cue Events should come right after segment info.");
160 return Dispatch(std::move(stream_data));
163Status WebVttToMp4Handler::OnSegmentInfo(
164 std::unique_ptr<StreamData> stream_data) {
166 DCHECK(stream_data->segment_info);
168 const auto& segment = stream_data->segment_info;
170 int64_t segment_start = segment->start_timestamp;
171 int64_t segment_duration = segment->duration;
172 int64_t segment_end = segment_start + segment_duration;
174 RETURN_IF_ERROR(DispatchCurrentSegment(segment_start, segment_end));
175 current_segment_.clear();
177 return Dispatch(std::move(stream_data));
180Status WebVttToMp4Handler::OnTextSample(
181 std::unique_ptr<StreamData> stream_data) {
183 DCHECK(stream_data->text_sample);
185 auto& sample = stream_data->text_sample;
189 if (sample->body().is_empty()) {
195 current_segment_.push_back(std::move(stream_data->text_sample));
199Status WebVttToMp4Handler::DispatchCurrentSegment(int64_t segment_start,
200 int64_t segment_end) {
203 std::list<const TextSample*> active;
207 int64_t section_start = segment_start;
210 auto actions = CreateActionList(segment_start, segment_end, current_segment_);
211 auto front = actions.begin();
216 while (section_start < segment_end) {
222 while (front != actions.end() && front->first <= section_start) {
223 auto& action = front->second;
225 switch (action.type) {
226 case DisplayActionType::ADD: {
227 active.push_back(action.sample);
230 case DisplayActionType::REMOVE: {
231 auto found = std::find(active.begin(), active.end(), action.sample);
232 DCHECK(found != active.end());
237 NOTIMPLEMENTED() <<
"Unsupported DisplayActionType "
238 <<
static_cast<int>(action.type);
249 int64_t section_end = front == actions.end() ? segment_end : front->first;
250 DCHECK_GT(section_end, section_start);
251 DCHECK_LE(section_end, segment_end);
252 RETURN_IF_ERROR(MergeDispatchSamples(section_start, section_end, active));
254 section_start = section_end;
257 DCHECK(front == actions.end()) <<
"We should have processed all actions.";
262Status WebVttToMp4Handler::MergeDispatchSamples(
265 const std::list<const TextSample*>& state) {
266 DCHECK_GT(end_time, start_time);
271 WriteSamples(state, &box_writer_);
273 WriteEmptySample(&box_writer_);
277 kTrackId, CreateMediaSample(box_writer_, start_time, end_time));
All the methods that are virtual are virtual for mocking.