Files
libopenshot/src/ChunkReader.cpp

222 lines
6.6 KiB
C++
Raw Normal View History

2013-07-31 12:45:47 -05:00
#include "../include/ChunkReader.h"
using namespace openshot;
ChunkReader::ChunkReader(string path, ChunkVersion chunk_version) throw(InvalidFile, InvalidJSON)
: path(path), chunk_size(24 * 3), is_open(false), version(chunk_version), local_reader(NULL)
2013-07-31 12:45:47 -05:00
{
// Init FileInfo struct (clear all values)
InitFileInfo();
// Check if folder exists?
if (!does_folder_exist(path))
// Raise exception
throw InvalidFile("Chunk folder could not be opened.", path);
// Init previous location
previous_location.number = 0;
previous_location.frame = 0;
2013-07-31 12:45:47 -05:00
// Open and Close the reader, to populate it's attributes (such as height, width, etc...)
Open();
Close();
}
// Check if folder path existing
bool ChunkReader::does_folder_exist(string path)
{
QDir dir(path.c_str());
return dir.exists();
}
// Load JSON meta data about this chunk folder
void ChunkReader::load_json()
{
// Load path of chunk folder
string json_path = QDir::cleanPath(QString(path.c_str()) + QDir::separator() + "info.json").toStdString();
stringstream json_string;
// Read the JSON file
ifstream myfile (json_path.c_str());
string line = "";
if (myfile.is_open())
{
while (myfile.good())
{
getline (myfile, line);
json_string << line;
}
myfile.close();
}
// Parse JSON string into JSON objects
Json::Value root;
Json::Reader reader;
bool success = reader.parse( json_string.str(), root );
if (!success)
// Raise exception
throw InvalidJSON("Chunk folder could not be opened.", path);
// Set info from the JSON objects
try
{
info.has_video = root["has_video"].asBool();
info.has_audio = root["has_audio"].asBool();
info.duration = root["duration"].asDouble();
info.file_size = atoll(root["file_size"].asString().c_str());
info.height = root["height"].asInt();
info.width = root["width"].asInt();
info.pixel_format = root["pixel_format"].asInt();
info.fps.num = root["fps"]["num"].asInt();
info.fps.den = root["fps"]["den"].asInt();
info.video_bit_rate = root["video_bit_rate"].asUInt();
info.pixel_ratio.num = root["pixel_ratio"]["num"].asInt();
info.pixel_ratio.den = root["pixel_ratio"]["den"].asInt();
info.display_ratio.num = root["display_ratio"]["num"].asInt();
info.display_ratio.den = root["display_ratio"]["den"].asInt();
info.vcodec = root["vcodec"].asString();
info.video_length = atoll(root["video_length"].asString().c_str());
info.video_stream_index = root["video_stream_index"].asInt();
info.video_timebase.num = root["video_timebase"]["num"].asInt();
info.video_timebase.den = root["video_timebase"]["den"].asInt();
info.interlaced_frame = root["interlaced_frame"].asBool();
info.top_field_first = root["top_field_first"].asBool();
info.acodec = root["acodec"].asString();
info.audio_bit_rate = root["audio_bit_rate"].asUInt();
info.sample_rate = root["sample_rate"].asUInt();
info.channels = root["channels"].asInt();
info.audio_stream_index = root["audio_stream_index"].asInt();
info.audio_timebase.num = root["audio_timebase"]["num"].asInt();
info.audio_timebase.den = root["audio_timebase"]["den"].asInt();
}
catch (exception e)
{
// Error parsing JSON (or missing keys)
throw InvalidJSON("JSON could not be parsed (or is invalid).", path);
}
}
// Find the location of a frame in a chunk
chunk_location ChunkReader::find_chunk_frame(int requested_frame)
{
// Determine which chunk contains this frame
int chunk_number = (requested_frame / chunk_size) + 1;
// Determine which frame in this chunk
int start_frame_of_chunk = (chunk_number - 1) * chunk_size;
int chunk_frame_number = requested_frame - start_frame_of_chunk;
// Prepare chunk location struct
chunk_location location = {chunk_number, chunk_frame_number};
return location;
}
2013-07-31 12:45:47 -05:00
// Open chunk folder or file
void ChunkReader::Open() throw(InvalidFile)
{
// Open reader if not already open
if (!is_open)
{
// parse JSON and load info.json file
load_json();
2013-07-31 12:45:47 -05:00
// Mark as "open"
is_open = true;
}
}
// Close image file
void ChunkReader::Close()
{
// Close all objects, if reader is 'open'
if (is_open)
{
// Mark as "closed"
is_open = false;
}
}
// get a formatted path of a specific chunk
string ChunkReader::get_chunk_path(int chunk_number, string folder, string extension)
2013-07-31 12:45:47 -05:00
{
// Create path of new chunk video
stringstream chunk_count_string;
chunk_count_string << chunk_number;
QString padded_count = "%1"; //chunk_count_string.str().c_str();
padded_count = padded_count.arg(chunk_count_string.str().c_str(), 6, '0');
if (folder.length() != 0 && extension.length() != 0)
// Return path with FOLDER and EXTENSION name
return QDir::cleanPath(QString(path.c_str()) + QDir::separator() + folder.c_str() + QDir::separator() + padded_count + extension.c_str()).toStdString();
else if (folder.length() == 0 && extension.length() != 0)
// Return path with NO FOLDER and EXTENSION name
return QDir::cleanPath(QString(path.c_str()) + QDir::separator() + padded_count + extension.c_str()).toStdString();
else if (folder.length() != 0 && extension.length() == 0)
// Return path with FOLDER and NO EXTENSION
return QDir::cleanPath(QString(path.c_str()) + QDir::separator() + folder.c_str()).toStdString();
}
// Get an openshot::Frame object for a specific frame number of this reader.
tr1::shared_ptr<Frame> ChunkReader::GetFrame(int requested_frame) throw(ReaderClosed, ChunkNotFound)
{
// Determine what chunk contains this frame
chunk_location location = find_chunk_frame(requested_frame);
// New Chunk (Close the old reader, and open the new one)
if (previous_location.number != location.number)
2013-07-31 12:45:47 -05:00
{
// Determine version of chunk
string folder_name = "";
switch (version)
{
case THUMBNAIL:
folder_name = "thumb";
break;
case PREVIEW:
folder_name = "preview";
break;
case FINAL:
folder_name = "final";
break;
}
2013-07-31 12:45:47 -05:00
// Load path of chunk video
string chunk_video_path = get_chunk_path(location.number, folder_name, ".webm");
2013-07-31 12:45:47 -05:00
// Close existing reader (if needed)
if (local_reader)
{
cout << "Close READER" << endl;
// Close and delete old reader
local_reader->Close();
delete local_reader;
}
try
{
cout << "Load READER: " << chunk_video_path << endl;
// Load new FFmpegReader
local_reader = new FFmpegReader(chunk_video_path);
local_reader->enable_seek = false; // disable seeking
local_reader->Open(); // open reader
} catch (InvalidFile)
{
// Invalid Chunk (possibly it is not found)
throw ChunkNotFound(path, requested_frame, location.number, location.frame);
}
// Set the new location
previous_location = location;
2013-07-31 12:45:47 -05:00
}
// Return the frame (from the current reader)
return local_reader->GetFrame(location.frame);
2013-07-31 12:45:47 -05:00
}