This repository has been archived on 2023-10-09. You can view files and clone it. You cannot open issues or pull requests or push a commit.
Files
blender-archive/source/blender/compositor/operations/COM_MovieDistortionOperation.h

171 lines
5.2 KiB
C++

/*
* Copyright 2011, Blender Foundation.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software Foundation,
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
* Contributor:
* Jeroen Bakker
* Monique Dewanchand
*/
#ifndef _COM_MovieDistortionOperation_h_
#define _COM_MovieDistortionOperation_h_
#include "COM_NodeOperation.h"
#include "DNA_movieclip_types.h"
#include "MEM_guardedalloc.h"
extern "C" {
#include "BKE_tracking.h"
#include "PIL_time.h"
}
#define COM_DISTORTIONCACHE_MAXSIZE 10
class DistortionCache {
private:
float m_k1;
float m_k2;
float m_k3;
float m_principal_x;
float m_principal_y;
float m_pixel_aspect;
int m_width;
int m_height;
int m_calibration_width;
int m_calibration_height;
bool m_inverted;
float *m_buffer;
int *m_bufferCalculated;
double timeLastUsage;
public:
DistortionCache(MovieClip *movieclip, int width, int height, int calibration_width, int calibration_height, bool inverted) {
this->m_k1 = movieclip->tracking.camera.k1;
this->m_k2 = movieclip->tracking.camera.k2;
this->m_k3 = movieclip->tracking.camera.k3;
this->m_principal_x = movieclip->tracking.camera.principal[0];
this->m_principal_y = movieclip->tracking.camera.principal[1];
this->m_pixel_aspect = movieclip->tracking.camera.pixel_aspect;
this->m_width = width;
this->m_height = height;
this->m_calibration_width = calibration_width;
this->m_calibration_height = calibration_height;
this->m_inverted = inverted;
this->m_bufferCalculated = (int *)MEM_callocN(sizeof(int) * this->m_width * this->m_height, __func__);
this->m_buffer = (float *)MEM_mallocN(sizeof(float) * this->m_width * this->m_height * 2, __func__);
this->updateLastUsage();
}
~DistortionCache() {
if (this->m_buffer) {
MEM_freeN(this->m_buffer);
this->m_buffer = NULL;
}
if (this->m_bufferCalculated) {
MEM_freeN(this->m_bufferCalculated);
this->m_bufferCalculated = NULL;
}
}
void updateLastUsage() {
this->timeLastUsage = PIL_check_seconds_timer();
}
inline double getTimeLastUsage() {
return this->timeLastUsage;
}
bool isCacheFor(MovieClip *movieclip, int width, int height, int calibration_width, int claibration_height, bool inverted) {
return this->m_k1 == movieclip->tracking.camera.k1 &&
this->m_k2 == movieclip->tracking.camera.k2 &&
this->m_k3 == movieclip->tracking.camera.k3 &&
this->m_principal_x == movieclip->tracking.camera.principal[0] &&
this->m_principal_y == movieclip->tracking.camera.principal[1] &&
this->m_pixel_aspect == movieclip->tracking.camera.pixel_aspect &&
this->m_inverted == inverted &&
this->m_width == width &&
this->m_height == height &&
this->m_calibration_width == this->m_calibration_width &&
this->m_calibration_height == this->m_calibration_height;
}
void getUV(MovieTracking *trackingData, int x, int y, float *u, float *v)
{
if (x < 0 || x >= this->m_width || y < 0 || y >= this->m_height) {
*u = x;
*v = y;
}
else {
int offset = y * this->m_width + x;
int offset2 = offset * 2;
if (!this->m_bufferCalculated[offset]) {
//float overscan = 0.0f;
float w = (float)this->m_width /* / (1 + overscan) */;
float h = (float)this->m_height /* / (1 + overscan) */;
float aspx = (float)w / this->m_calibration_width;
float aspy = (float)h / this->m_calibration_height;
float in[2];
float out[2];
in[0] = (x /* - 0.5 * overscan * w */) / aspx;
in[1] = (y /* - 0.5 * overscan * h */) / aspy / this->m_pixel_aspect;
if (this->m_inverted) {
BKE_tracking_undistort_v2(trackingData, in, out);
}
else {
BKE_tracking_distort_v2(trackingData, in, out);
}
this->m_buffer[offset2] = out[0] * aspx /* + 0.5 * overscan * w */;
this->m_buffer[offset2 + 1] = (out[1] * aspy /* + 0.5 * overscan * h */) * this->m_pixel_aspect;
this->m_bufferCalculated[offset] = 1;
}
*u = this->m_buffer[offset2];
*v = this->m_buffer[offset2 + 1];
}
}
};
class MovieDistortionOperation : public NodeOperation {
private:
DistortionCache *m_cache;
SocketReader *m_inputOperation;
MovieClip *m_movieClip;
protected:
bool m_distortion;
int m_framenumber;
public:
MovieDistortionOperation(bool distortion);
bool determineDependingAreaOfInterest(rcti *input, ReadBufferOperation *readOperation, rcti *output);
void executePixel(float output[4], float x, float y, PixelSampler sampler);
void initExecution();
void deinitExecution();
void setMovieClip(MovieClip *clip) { this->m_movieClip = clip; }
void setFramenumber(int framenumber) { this->m_framenumber = framenumber; }
};
void deintializeDistortionCache(void);
#endif