* Added support for non-raw files to the film negative tool. The tool is now under the "Color" tab. Moved the entire filmneg code downstream, right after input profile conversion. Usage changes a bit: White Balance must be set to the _backlight_ color temperature. Added two more sliders to color-balance the picture after negative inversion. Legacy inversion method kept for compatibility with processing profiles saved by RT v5.7 or 5.8 (only if Film Negative was enabled). Should be removed in a future version. There is still an issue with DCP profiles that need the look table to be active to work properly. Using a simple matrix input profile (or just camera standard) is recommended for now. * The user can now choose to perform inversion before or after input colorspace conversion. Seamless backwards compatibility with previous processing profiles; upgrading from a previous version now gives an (almost) identical output as before. Generalised the concept of film base values: the processing profile now contains a pair of RGB triplets, "reference input" and "reference output", which makes it much more straightforward to compute the output multipliers. * Added support for `RGB` data type to putToKeyFile, removed the now unused `RGB::toVector()` method. Some cleanup. * Spot balance picker now stays active indefinitely. Can be disabled by right-clicking, too. * Removed film negative from `filterRawRefresh` condition, since the new version does not require raw rendering anymore. * The Output Level slider is now exponential, so it should feel more familiar to use (similar to the exposure compensation slider). * Removed old `RedBase`, `GreenBase`, `BlueBase` keys from the PP3 file after params upgrade. Keys are only kept if the file only undergoes batch edit (hence no params upgrade was done). * Made the balance sliders exponential and centered at zero. Now they should be a bit smoother and possibly more user-friendly. * Changed adjusters' step to more useful values, they were too fine-grained and hard to adjust using the +/- spinbutton. Increased max ratio value from 3 to 5, as i found an old negative needing a very high blue channel exponent. * Added an initial processing profile for film negative inversion, to be provided as a bundled profile. * Removed Output Level and balance sliders when in batch mode: since the effect of these sliders is dependent on the reference input values, those values needed to be copied as well. And, touching any of the sliders needed to flag all three as dirty. All this felt more confusing than useful. It should be sufficient (and clearer) to copy/paste the params, and then fine-tune the balance on individual pictures when needed. * Set bayer demosaic method to RCD in the bundled "Film Negative" processing profile. RCD seems to play a bit better with Capture Sharpening in the presence of film grain, compared to Amaze. This will favor new users starting with all-defaults settings, hence having Capture Sharpening enabled by default. * Removed incorrect "contrast" term from the "Reference exponent" label. This parameter adjusts the image _gamma_, not its contrast.
310 lines
18 KiB
C++
310 lines
18 KiB
C++
/*
|
|
* This file is part of RawTherapee.
|
|
*
|
|
* Copyright (c) 2004-2010 Gabor Horvath <hgabor@rawtherapee.com>
|
|
*
|
|
* RawTherapee 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 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* RawTherapee 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 RawTherapee. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
#pragma once
|
|
|
|
#include <array>
|
|
#include <iostream>
|
|
#include <memory>
|
|
|
|
#include "array2D.h"
|
|
#include "colortemp.h"
|
|
#include "iimage.h"
|
|
#include "imagesource.h"
|
|
#include "procparams.h"
|
|
|
|
#define HR_SCALE 2
|
|
|
|
namespace rtengine
|
|
{
|
|
class PixelsMap;
|
|
class RawImage;
|
|
class DiagonalCurve;
|
|
class RetinextransmissionCurve;
|
|
class RetinexgaintransmissionCurve;
|
|
|
|
class RawImageSource final : public ImageSource
|
|
{
|
|
private:
|
|
static DiagonalCurve *phaseOneIccCurve;
|
|
static DiagonalCurve *phaseOneIccCurveInv;
|
|
static LUTf invGrad; // for fast_demosaic
|
|
static LUTf initInvGrad ();
|
|
static void colorSpaceConversion_ (Imagefloat* im, const procparams::ColorManagementParams& cmp, const ColorTemp &wb, double pre_mul[3], cmsHPROFILE embedded, cmsHPROFILE camprofile, double cam[3][3], const std::string &camName);
|
|
int defTransform (int tran);
|
|
|
|
protected:
|
|
MyMutex getImageMutex; // locks getImage
|
|
|
|
int W, H;
|
|
ColorTemp camera_wb;
|
|
ProgressListener* plistener;
|
|
float scale_mul[4]; // multiplier for each color
|
|
float c_black[4]; // copy of cblack Dcraw for black level
|
|
float c_white[4];
|
|
float cblacksom[4];
|
|
float ref_pre_mul[4];
|
|
double refwb_red;
|
|
double refwb_green;
|
|
double refwb_blue;
|
|
double rgb_cam[3][3];
|
|
double cam_rgb[3][3];
|
|
double xyz_cam[3][3];
|
|
double cam_xyz[3][3];
|
|
bool fuji;
|
|
bool d1x;
|
|
int border;
|
|
float chmax[4], hlmax[4], clmax[4];
|
|
double initialGain; // initial gain calculated after scale_colors
|
|
double camInitialGain;
|
|
double defGain;
|
|
cmsHPROFILE camProfile;
|
|
bool rgbSourceModified;
|
|
|
|
RawImage* ri; // Copy of raw pixels, NOT corrected for initial gain, blackpoint etc.
|
|
RawImage* riFrames[6] = {nullptr};
|
|
unsigned int currFrame = 0;
|
|
unsigned int numFrames = 0;
|
|
int flatFieldAutoClipValue = 0;
|
|
array2D<float> rawData; // holds preprocessed pixel values, rowData[i][j] corresponds to the ith row and jth column
|
|
array2D<float> *rawDataFrames[6] = {nullptr};
|
|
array2D<float> *rawDataBuffer[5] = {nullptr};
|
|
|
|
// the interpolated green plane:
|
|
array2D<float> green;
|
|
array2D<float> greenloc;
|
|
// the interpolated red plane:
|
|
array2D<float> red;
|
|
array2D<float> redloc;
|
|
// the interpolated blue plane:
|
|
array2D<float> blue;
|
|
array2D<float> blueloc;
|
|
array2D<float>* greenCache;
|
|
// the interpolated red plane:
|
|
array2D<float>* redCache;
|
|
// the interpolated blue plane:
|
|
array2D<float>* blueCache;
|
|
bool rawDirty;
|
|
float psRedBrightness[4];
|
|
float psGreenBrightness[4];
|
|
float psBlueBrightness[4];
|
|
|
|
std::vector<double> histMatchingCache;
|
|
const std::unique_ptr<procparams::ColorManagementParams> histMatchingParams;
|
|
|
|
void processFalseColorCorrectionThread(Imagefloat* im, array2D<float> &rbconv_Y, array2D<float> &rbconv_I, array2D<float> &rbconv_Q, array2D<float> &rbout_I, array2D<float> &rbout_Q, const int row_from, const int row_to);
|
|
void hlRecovery(const std::string &method, float* red, float* green, float* blue, int width, float* hlmax);
|
|
void transformRect(const PreviewProps &pp, int tran, int &sx1, int &sy1, int &width, int &height, int &fw);
|
|
void transformPosition(int x, int y, int tran, int& tx, int& ty);
|
|
void ItcWB(bool extra, double &tempref, double &greenref, double &tempitc, double &greenitc, float &studgood, array2D<float> &redloc, array2D<float> &greenloc, array2D<float> &blueloc, int bfw, int bfh, double &avg_rm, double &avg_gm, double &avg_bm, const procparams::ColorManagementParams &cmp, const procparams::RAWParams &raw, const procparams::WBParams & wbpar);
|
|
|
|
unsigned FC(int row, int col) const;
|
|
inline void getRowStartEnd (int x, int &start, int &end);
|
|
static void getProfilePreprocParams(cmsHPROFILE in, float& gammafac, float& lineFac, float& lineSum);
|
|
|
|
public:
|
|
RawImageSource ();
|
|
~RawImageSource () override;
|
|
|
|
int load(const Glib::ustring &fname) override { return load(fname, false); }
|
|
int load(const Glib::ustring &fname, bool firstFrameOnly);
|
|
void preprocess (const procparams::RAWParams &raw, const procparams::LensProfParams &lensProf, const procparams::CoarseTransformParams& coarse, bool prepareDenoise = true) override;
|
|
void demosaic (const procparams::RAWParams &raw, bool autoContrast, double &contrastThreshold, bool cache = false) override;
|
|
void retinex (const procparams::ColorManagementParams& cmp, const procparams::RetinexParams &deh, const procparams::ToneCurveParams& Tc, LUTf & cdcurve, LUTf & mapcurve, const RetinextransmissionCurve & dehatransmissionCurve, const RetinexgaintransmissionCurve & dehagaintransmissionCurve, multi_array2D<float, 4> &conversionBuffer, bool dehacontlutili, bool mapcontlutili, bool useHsl, float &minCD, float &maxCD, float &mini, float &maxi, float &Tmean, float &Tsigma, float &Tmin, float &Tmax, LUTu &histLRETI) override;
|
|
void retinexPrepareCurves (const procparams::RetinexParams &retinexParams, LUTf &cdcurve, LUTf &mapcurve, RetinextransmissionCurve &retinextransmissionCurve, RetinexgaintransmissionCurve &retinexgaintransmissionCurve, bool &retinexcontlutili, bool &mapcontlutili, bool &useHsl, LUTu & lhist16RETI, LUTu & histLRETI) override;
|
|
void retinexPrepareBuffers (const procparams::ColorManagementParams& cmp, const procparams::RetinexParams &retinexParams, multi_array2D<float, 4> &conversionBuffer, LUTu &lhist16RETI) override;
|
|
void flush () override;
|
|
void HLRecovery_Global (const procparams::ToneCurveParams &hrp) override;
|
|
void refinement(int PassCount);
|
|
void setBorder(unsigned int rawBorder) override {border = rawBorder;}
|
|
bool isRGBSourceModified() const override
|
|
{
|
|
return rgbSourceModified; // tracks whether cached rgb output of demosaic has been modified
|
|
}
|
|
|
|
void processFlatField(const procparams::RAWParams &raw, const RawImage *riFlatFile, const float black[4]);
|
|
void copyOriginalPixels(const procparams::RAWParams &raw, RawImage *ri, RawImage *riDark, RawImage *riFlatFile, array2D<float> &rawData );
|
|
void scaleColors (int winx, int winy, int winw, int winh, const procparams::RAWParams &raw, array2D<float> &rawData); // raw for cblack
|
|
void WBauto(double &tempref, double &greenref, array2D<float> &redloc, array2D<float> &greenloc, array2D<float> &blueloc, int bfw, int bfh, double &avg_rm, double &avg_gm, double &avg_bm, double &tempitc, double &greenitc, float &studgood, bool &twotimes, const procparams::WBParams & wbpar, int begx, int begy, int yEn, int xEn, int cx, int cy, const procparams::ColorManagementParams &cmp, const procparams::RAWParams &raw) override;
|
|
void getAutoWBMultipliersitc(double &tempref, double &greenref, double &tempitc, double &greenitc, float &studgood, int begx, int begy, int yEn, int xEn, int cx, int cy, int bf_h, int bf_w, double &rm, double &gm, double &bm, const procparams::WBParams & wbpar, const procparams::ColorManagementParams &cmp, const procparams::RAWParams &raw) override;
|
|
void getrgbloc(int begx, int begy, int yEn, int xEn, int cx, int cy, int bf_h, int bf_w) override;
|
|
|
|
void getWBMults (const ColorTemp &ctemp, const procparams::RAWParams &raw, std::array<float, 4>& scale_mul, float &autoGainComp, float &rm, float &gm, float &bm) const override;
|
|
void getImage (const ColorTemp &ctemp, int tran, Imagefloat* image, const PreviewProps &pp, const procparams::ToneCurveParams &hrp, const procparams::RAWParams &raw) override;
|
|
eSensorType getSensorType () const override;
|
|
bool isMono () const override;
|
|
ColorTemp getWB () const override
|
|
{
|
|
return camera_wb;
|
|
}
|
|
void getAutoWBMultipliers (double &rm, double &gm, double &bm) override;
|
|
ColorTemp getSpotWB (std::vector<Coord2D> &red, std::vector<Coord2D> &green, std::vector<Coord2D> &blue, int tran, double equal) override;
|
|
bool isWBProviderReady () override
|
|
{
|
|
return rawData;
|
|
}
|
|
|
|
double getDefGain () const override
|
|
{
|
|
return defGain;
|
|
}
|
|
|
|
void getFullSize (int& w, int& h, int tr = TR_NONE) override;
|
|
void getSize (const PreviewProps &pp, int& w, int& h) override;
|
|
int getRotateDegree() const override;
|
|
|
|
ImageMatrices* getImageMatrices () override
|
|
{
|
|
return &imatrices;
|
|
}
|
|
bool isRAW() const override
|
|
{
|
|
return true;
|
|
}
|
|
|
|
void setProgressListener (ProgressListener* pl) override
|
|
{
|
|
plistener = pl;
|
|
}
|
|
void getAutoExpHistogram (LUTu & histogram, int& histcompr) override;
|
|
void getRAWHistogram (LUTu & histRedRaw, LUTu & histGreenRaw, LUTu & histBlueRaw) override;
|
|
void getAutoMatchedToneCurve(const procparams::ColorManagementParams &cp, std::vector<double> &outCurve) override;
|
|
DCPProfile *getDCP(const procparams::ColorManagementParams &cmp, DCPProfileApplyState &as) override;
|
|
|
|
void convertColorSpace(Imagefloat* image, const procparams::ColorManagementParams &cmp, const ColorTemp &wb) override;
|
|
static bool findInputProfile(Glib::ustring inProfile, cmsHPROFILE embedded, std::string camName, DCPProfile **dcpProf, cmsHPROFILE& in);
|
|
static void colorSpaceConversion(Imagefloat* im, const procparams::ColorManagementParams& cmp, const ColorTemp &wb, double pre_mul[3], cmsHPROFILE embedded, cmsHPROFILE camprofile, double cam[3][3], const std::string &camName)
|
|
{
|
|
colorSpaceConversion_(im, cmp, wb, pre_mul, embedded, camprofile, cam, camName);
|
|
}
|
|
static void inverse33(const double (*coeff)[3], double (*icoeff)[3]);
|
|
|
|
void MSR(float** luminance, float **originalLuminance, float **exLuminance, const LUTf& mapcurve, bool mapcontlutili, int width, int height, const procparams::RetinexParams &deh, const RetinextransmissionCurve & dehatransmissionCurve, const RetinexgaintransmissionCurve & dehagaintransmissionCurve, float &minCD, float &maxCD, float &mini, float &maxi, float &Tmean, float &Tsigma, float &Tmin, float &Tmax);
|
|
void HLRecovery_inpaint (float** red, float** green, float** blue) override;
|
|
static void HLRecovery_Luminance (float* rin, float* gin, float* bin, float* rout, float* gout, float* bout, int width, float maxval);
|
|
static void HLRecovery_CIELab (float* rin, float* gin, float* bin, float* rout, float* gout, float* bout, int width, float maxval, double cam[3][3], double icam[3][3]);
|
|
static void HLRecovery_blend (float* rin, float* gin, float* bin, int width, float maxval, float* hlmax);
|
|
static void init ();
|
|
static void cleanup ();
|
|
void setCurrentFrame(unsigned int frameNum) override {
|
|
if (numFrames == 2 && frameNum == 2) { // special case for averaging of two frames
|
|
currFrame = frameNum;
|
|
ri = riFrames[0];
|
|
} else {
|
|
currFrame = std::min(numFrames - 1, frameNum);
|
|
ri = riFrames[currFrame];
|
|
}
|
|
}
|
|
int getFrameCount() override {return numFrames;}
|
|
int getFlatFieldAutoClipValue() override {return flatFieldAutoClipValue;}
|
|
|
|
class GreenEqulibrateThreshold {
|
|
public:
|
|
explicit GreenEqulibrateThreshold(float thresh): thresh_(thresh) {}
|
|
virtual ~GreenEqulibrateThreshold() {}
|
|
virtual float operator()(int row, int column) const { return thresh_; }
|
|
protected:
|
|
const float thresh_;
|
|
};
|
|
|
|
class CFALineDenoiseRowBlender {
|
|
public:
|
|
virtual ~CFALineDenoiseRowBlender() {}
|
|
virtual float operator()(int row) const { return 1.f; }
|
|
};
|
|
|
|
protected:
|
|
typedef unsigned short ushort;
|
|
void processFalseColorCorrection(Imagefloat* i, const int steps);
|
|
inline void convert_row_to_YIQ(const float* const r, const float* const g, const float* const b, float* Y, float* I, float* Q, const int W);
|
|
inline void convert_row_to_RGB(float* r, float* g, float* b, const float* const Y, const float* const I, const float* const Q, const int W);
|
|
inline void convert_to_RGB(float &r, float &g, float &b, const float Y, const float I, const float Q);
|
|
|
|
inline void interpolate_row_g (float* agh, float* agv, int i);
|
|
inline void interpolate_row_rb (float* ar, float* ab, float* pg, float* cg, float* ng, int i);
|
|
inline void interpolate_row_rb_mul_pp (const array2D<float> &rawData, float* ar, float* ab, float* pg, float* cg, float* ng, int i, float r_mul, float g_mul, float b_mul, int x1, int width, int skip);
|
|
|
|
float* CA_correct_RT(
|
|
bool autoCA,
|
|
size_t autoIterations,
|
|
double cared,
|
|
double cablue,
|
|
bool avoidColourshift,
|
|
array2D<float> &rawData,
|
|
double* fitParamsTransfer,
|
|
bool fitParamsIn,
|
|
bool fitParamsOut,
|
|
float* buffer,
|
|
bool freeBuffer,
|
|
size_t chunkSize = 1,
|
|
bool measure = false
|
|
);
|
|
void ddct8x8s(int isgn, float a[8][8]);
|
|
|
|
int interpolateBadPixelsBayer(const PixelsMap &bitmapBads, array2D<float> &rawData);
|
|
int interpolateBadPixelsNColours(const PixelsMap &bitmapBads, int colours);
|
|
int interpolateBadPixelsXtrans(const PixelsMap &bitmapBads);
|
|
int findHotDeadPixels(PixelsMap &bpMap, float thresh, bool findHotPixels, bool findDeadPixels) const;
|
|
int findZeroPixels(PixelsMap &bpMap) const;
|
|
void cfa_linedn (float linenoiselevel, bool horizontal, bool vertical, const CFALineDenoiseRowBlender &rowblender);//Emil's line denoise
|
|
|
|
void green_equilibrate_global(array2D<float> &rawData);
|
|
void green_equilibrate (const GreenEqulibrateThreshold &greenthresh, array2D<float> &rawData);//Emil's green equilibration
|
|
|
|
void nodemosaic(bool bw);
|
|
void eahd_demosaic();
|
|
void hphd_demosaic();
|
|
void vng4_demosaic(const array2D<float> &rawData, array2D<float> &red, array2D<float> &green, array2D<float> &blue);
|
|
void igv_interpolate(int winw, int winh);
|
|
void lmmse_interpolate_omp(int winw, int winh, const array2D<float> &rawData, array2D<float> &red, array2D<float> &green, array2D<float> &blue, int iterations);
|
|
void amaze_demosaic_RT(int winx, int winy, int winw, int winh, const array2D<float> &rawData, array2D<float> &red, array2D<float> &green, array2D<float> &blue, size_t chunkSize = 1, bool measure = false);//Emil's code for AMaZE
|
|
void dual_demosaic_RT(bool isBayer, const procparams::RAWParams &raw, int winw, int winh, const array2D<float> &rawData, array2D<float> &red, array2D<float> &green, array2D<float> &blue, double &contrast, bool autoContrast = false);
|
|
void fast_demosaic();//Emil's code for fast demosaicing
|
|
void dcb_demosaic(int iterations, bool dcb_enhance);
|
|
void ahd_demosaic();
|
|
void rcd_demosaic(size_t chunkSize = 1, bool measure = false);
|
|
void border_interpolate(int winw, int winh, int lborders, const array2D<float> &rawData, array2D<float> &red, array2D<float> &green, array2D<float> &blue);
|
|
void dcb_initTileLimits(int &colMin, int &rowMin, int &colMax, int &rowMax, int x0, int y0, int border);
|
|
void fill_raw(float (*cache)[3], int x0, int y0, float** rawData);
|
|
void fill_border(float (*cache)[3], int border, int x0, int y0);
|
|
void copy_to_buffer(float (*image2)[2], float (*image)[3]);
|
|
void dcb_hid(float (*image)[3], int x0, int y0);
|
|
void dcb_color(float (*image)[3], int x0, int y0);
|
|
void dcb_hid2(float (*image)[3], int x0, int y0);
|
|
void dcb_map(float (*image)[3], uint8_t *map, int x0, int y0);
|
|
void dcb_correction(float (*image)[3], uint8_t *map, int x0, int y0);
|
|
void dcb_pp(float (*image)[3], int x0, int y0);
|
|
void dcb_correction2(float (*image)[3], uint8_t *map, int x0, int y0);
|
|
void restore_from_buffer(float (*image)[3], float (*image2)[2]);
|
|
void dcb_refinement(float (*image)[3], uint8_t *map, int x0, int y0);
|
|
void dcb_color_full(float (*image)[3], int x0, int y0, float (*chroma)[2]);
|
|
void cielab(const float (*rgb)[3], float* l, float* a, float *b, const int width, const int height, const int labWidth, const float xyz_cam[3][3]);
|
|
void xtransborder_interpolate (int border, array2D<float> &red, array2D<float> &green, array2D<float> &blue);
|
|
void xtrans_interpolate (const int passes, const bool useCieLab, size_t chunkSize = 1, bool measure = false);
|
|
void fast_xtrans_interpolate (const array2D<float> &rawData, array2D<float> &red, array2D<float> &green, array2D<float> &blue);
|
|
void fast_xtrans_interpolate_blend (const float* const * blend, const array2D<float> &rawData, array2D<float> &red, array2D<float> &green, array2D<float> &blue);
|
|
void pixelshift(int winx, int winy, int winw, int winh, const procparams::RAWParams &rawParams, unsigned int frame, const std::string &make, const std::string &model, float rawWpCorrection);
|
|
void bayer_bilinear_demosaic(const float *const * blend, const array2D<float> &rawData, array2D<float> &red, array2D<float> &green, array2D<float> &blue);
|
|
void hflip (Imagefloat* im);
|
|
void vflip (Imagefloat* im);
|
|
void getRawValues(int x, int y, int rotate, int &R, int &G, int &B) override;
|
|
void captureSharpening(const procparams::CaptureSharpeningParams &sharpeningParams, bool showMask, double &conrastThreshold, double &radius) override;
|
|
};
|
|
|
|
}
|