Files
libopenshot/src/effects/Stabilizer.cpp

232 lines
7.1 KiB
C++
Raw Normal View History

2020-07-08 22:31:26 -03:00
/**
* @file
* @brief Source file for Stabilizer effect class
* @author Jonathan Thomas <jonathan@openshot.org>
* @author Brenno Caldato <brenno.caldato@outlook.com>
2020-07-08 22:31:26 -03:00
*
* @ref License
*/
// Copyright (c) 2008-2019 OpenShot Studios, LLC
//
// SPDX-License-Identifier: LGPL-3.0-or-later
2020-07-08 22:31:26 -03:00
#include <fstream>
#include <iomanip>
#include <iostream>
2020-10-20 01:04:59 -03:00
#include "effects/Stabilizer.h"
#include "Exceptions.h"
#include "stabilizedata.pb.h"
#include <google/protobuf/util/time_util.h>
2020-07-08 22:31:26 -03:00
using namespace std;
2020-07-08 22:31:26 -03:00
using namespace openshot;
using google::protobuf::util::TimeUtil;
2020-07-08 22:31:26 -03:00
/// Blank constructor, useful when using Json to load the effect properties
Stabilizer::Stabilizer(std::string clipStabilizedDataPath):protobuf_data_path(clipStabilizedDataPath)
{
2020-07-08 22:31:26 -03:00
// Init effect properties
init_effect_details();
// Tries to load the stabilization data from protobuf
LoadStabilizedData(clipStabilizedDataPath);
}
// Default constructor
Stabilizer::Stabilizer()
2020-07-08 22:31:26 -03:00
{
// Init effect properties
init_effect_details();
}
// Init effect settings
void Stabilizer::init_effect_details()
{
2020-07-08 22:31:26 -03:00
/// Initialize the values of the EffectInfo struct.
InitEffectInfo();
/// Set the effect info
info.class_name = "Stabilizer";
info.name = "Stabilizer";
info.description = "Stabilize video clip to remove undesired shaking and jitter.";
info.has_audio = false;
info.has_video = true;
protobuf_data_path = "";
2020-07-22 14:05:19 -03:00
zoom = 1.0;
2020-07-08 22:31:26 -03:00
}
// This method is required for all derived classes of EffectBase, and returns a
// modified openshot::Frame object
std::shared_ptr<Frame> Stabilizer::GetFrame(std::shared_ptr<Frame> frame, int64_t frame_number)
{
// Grab OpenCV Mat image
cv::Mat frame_image = frame->GetImageCV();
// If frame is NULL, return itself
if(!frame_image.empty()){
2020-07-08 22:31:26 -03:00
// Check if track data exists for the requested frame
if(transformationData.find(frame_number) != transformationData.end()){
2020-07-22 14:05:19 -03:00
float zoom_value = zoom.GetValue(frame_number);
// Create empty rotation matrix
cv::Mat T(2,3,CV_64F);
2020-07-08 22:31:26 -03:00
// Set rotation matrix values
T.at<double>(0,0) = cos(transformationData[frame_number].da);
T.at<double>(0,1) = -sin(transformationData[frame_number].da);
T.at<double>(1,0) = sin(transformationData[frame_number].da);
T.at<double>(1,1) = cos(transformationData[frame_number].da);
2020-08-01 14:06:38 -03:00
T.at<double>(0,2) = transformationData[frame_number].dx * frame_image.size().width;
T.at<double>(1,2) = transformationData[frame_number].dy * frame_image.size().height;
// Apply rotation matrix to image
cv::Mat frame_stabilized;
cv::warpAffine(frame_image, frame_stabilized, T, frame_image.size());
// Scale up the image to remove black borders
cv::Mat T_scale = cv::getRotationMatrix2D(cv::Point2f(frame_stabilized.cols/2, frame_stabilized.rows/2), 0, zoom_value);
cv::warpAffine(frame_stabilized, frame_stabilized, T_scale, frame_stabilized.size());
frame_image = frame_stabilized;
}
}
// Set stabilized image to frame
// If the input image is NULL or doesn't have tracking data, it's returned as it came
frame->SetImageCV(frame_image);
2020-07-08 22:31:26 -03:00
return frame;
}
// Load protobuf data file
2020-07-08 22:31:26 -03:00
bool Stabilizer::LoadStabilizedData(std::string inputFilePath){
using std::ios;
// Create stabilization message
2021-01-13 13:36:15 -05:00
pb_stabilize::Stabilization stabilizationMessage;
2020-07-08 22:31:26 -03:00
// Read the existing tracker message.
std::fstream input(inputFilePath, ios::in | ios::binary);
2020-07-08 22:31:26 -03:00
if (!stabilizationMessage.ParseFromIstream(&input)) {
std::cerr << "Failed to parse protobuf message." << std::endl;
2020-07-08 22:31:26 -03:00
return false;
}
// Make sure the data maps are empty
2020-07-08 22:31:26 -03:00
transformationData.clear();
trajectoryData.clear();
// Iterate over all frames of the saved message and assign to the data maps
for (size_t i = 0; i < stabilizationMessage.frame_size(); i++) {
// Create stabilization message
2021-01-13 13:36:15 -05:00
const pb_stabilize::Frame& pbFrameData = stabilizationMessage.frame(i);
// Load frame number
size_t id = pbFrameData.id();
2020-07-08 22:31:26 -03:00
// Load camera trajectory data
2020-07-08 22:31:26 -03:00
float x = pbFrameData.x();
float y = pbFrameData.y();
float a = pbFrameData.a();
// Assign data to trajectory map
trajectoryData[i] = EffectCamTrajectory(x,y,a);
2020-07-08 22:31:26 -03:00
// Load transformation data
2020-07-08 22:31:26 -03:00
float dx = pbFrameData.dx();
float dy = pbFrameData.dy();
float da = pbFrameData.da();
// Assing data to transformation map
transformationData[id] = EffectTransformParam(dx,dy,da);
2020-07-08 22:31:26 -03:00
}
// Delete all global objects allocated by libprotobuf.
google::protobuf::ShutdownProtobufLibrary();
return true;
}
// Generate JSON string of this object
std::string Stabilizer::Json() const {
// Return formatted string
return JsonValue().toStyledString();
}
// Generate Json::Value for this object
Json::Value Stabilizer::JsonValue() const {
// Create root json object
Json::Value root = EffectBase::JsonValue(); // get parent properties
root["type"] = info.class_name;
root["protobuf_data_path"] = protobuf_data_path;
2020-07-22 14:05:19 -03:00
root["zoom"] = zoom.JsonValue();
2020-07-08 22:31:26 -03:00
// return JsonValue
return root;
}
// Load JSON string into this object
void Stabilizer::SetJson(const std::string value) {
// Parse JSON string into JSON objects
try
{
const Json::Value root = openshot::stringToJson(value);
// Set all values that match
2020-07-08 22:31:26 -03:00
SetJsonValue(root);
}
catch (const std::exception& e)
{
// Error parsing JSON (or missing keys)
throw InvalidJSON("JSON is invalid (missing keys or invalid data types)");
}
}
// Load Json::Value into this object
void Stabilizer::SetJsonValue(const Json::Value root) {
// Set parent data
EffectBase::SetJsonValue(root);
// Set data from Json (if key is found)
if (!root["protobuf_data_path"].isNull()){
protobuf_data_path = (root["protobuf_data_path"].asString());
if(!LoadStabilizedData(protobuf_data_path)){
std::cout<<"Invalid protobuf data path";
protobuf_data_path = "";
}
}
2020-07-22 14:05:19 -03:00
if(!root["zoom"].isNull())
zoom.SetJsonValue(root["zoom"]);
2020-07-08 22:31:26 -03:00
}
// Get all properties for a specific frame
std::string Stabilizer::PropertiesJSON(int64_t requested_frame) const {
// Generate JSON properties list
Json::Value root;
root["id"] = add_property_json("ID", 0.0, "string", Id(), NULL, -1, -1, true, requested_frame);
root["position"] = add_property_json("Position", Position(), "float", "", NULL, 0, 1000 * 60 * 30, false, requested_frame);
root["layer"] = add_property_json("Track", Layer(), "int", "", NULL, 0, 20, false, requested_frame);
root["start"] = add_property_json("Start", Start(), "float", "", NULL, 0, 1000 * 60 * 30, false, requested_frame);
root["end"] = add_property_json("End", End(), "float", "", NULL, 0, 1000 * 60 * 30, false, requested_frame);
root["duration"] = add_property_json("Duration", Duration(), "float", "", NULL, 0, 1000 * 60 * 30, true, requested_frame);
2020-07-22 14:05:19 -03:00
root["zoom"] = add_property_json("Zoom", zoom.GetValue(requested_frame), "float", "", &zoom, 0.0, 2.0, false, requested_frame);
// Set the parent effect which properties this effect will inherit
2021-03-26 21:10:10 -03:00
root["parent_effect_id"] = add_property_json("Parent", 0.0, "string", info.parent_effect_id, NULL, -1, -1, false, requested_frame);
2020-07-08 22:31:26 -03:00
// Return formatted string
return root.toStyledString();
}