* Change estimchrom estim hue final * Itcw observer checkbox * 8 spectral colors * 11 spectral colors * 6 spectral colors * Small change to correction green * Fixed various bad behavior - settings itcwb_deltaspec * 4 spectral colors green * 3 green spectral colors * 3 spectral colors green * 5 spectral colors red green * 3 spectral colors red green * 3 spectral colrs green red * Chnage slider choice deltaE - to size color patch * display more info in console * Display more info in console - comment code * Clean - comment code * 8 spectral colors green red * 8 spectral colors blue * 8 spectral colors blue * 3 spectral colors * 6 spectral colors * 4 spectal colors * 2 spectral colors * Eliminate high Y - increse temp iteration - recenter selection inside patch * compatibility 5.9 * Modify dislay and place selected dats in patch * Dispaly in console hue chroma image selection * First optimization * optimization code - clean unused variables * Remove sorted in chroma order - some others changes * 5.9 compatibility * Clean and optimize code * Clean code - more display info in console * More info in console - set itcwb_minsize in option * Display patch chroma in GUI * Change some default settings * Minsize patch GUI * Display GUI patch size * Chroma patch variations * Display in GUI min and max dats found in patch * Some adjustments * Remove too low datas in patch * Various improvment - ponderate * Clean code colortemp comment spectral datas * Comment code - correc gren * Comment and refine code * Fixed bug in improcoordinator * fixes a malfunction of AWB bias in improcoordinator * Various improvments - clean code * 2 improvments green and find patch * Allows parametrind ponder and Ypurple in options * Update autowb - filter on magenta purple * Clean and optimize code * Read colors in GUI * Denoise median 5x5 image datas before Itcwb * Change to GUI * Replace median5x5 by 2 median3x3 - clean GUI code * Work around to recalculate wbauto * Remove work around - change params->wb related change * Disabled low_sampling 5.9 * First try for compatibility Low sampling 5.9 * Re-enable WB autogrey * Change reference image to autogrey if camera settings probably out * Clean code aand adjust settings camera out * Format astylert rawimagesource.cc - various small adjustments * Small correction * Small green correction * Change default settings - correction to autowb bias in improccoordinator * Save provisory work * Save provisory work 2 * Try to find good tempitc with iterate when temptitc very different 5000K * Change 2 default settings * Work around awb temp bias * Limit work around bias - small other changes * 3 spectral colors deep blue * 4 spectral color deep blue * 4 color spectral yellow * 5 spectral colors neutral red-magenta * 5 spectral color neutral red magenta * Change threshold tempitc * 5.9 compatibility * Work around bad claculation Planck's formula * Fixed bad behavior when using blackbody spectal * 6 spectral color - checkbutton remove 2 passes * Fixed various bugs * Format rawimagesource.cc * Various improvment before to find the good selection * First new optimization (perhaps?) * Display GUI patc deltaE * Various optimzation and tooltip * Default observer 2°- Simplify GUI - improve display GUI - remove too low numbers datas * 4 spectral colors - change 2 settings * Change calculation threshold 2 passes * 6 spectral colors magenta * 13 spectral colors * Improvment GUI display * remove checkbutton no_purple * 2 spectral colors - remove in GUI low sampling 5.9 * Change default setting display in console * Change behavior when temp near stdA * 4 spectral colors - settings Itcwb_tempstdA * Gamut control on image histogram * Clean code and format rawimagesource.cc * Comment and format colortemp.cc * Remove unused fileds from GUI - change tooltips * Add tooltips * Various changes * Change settings when camera temperature is out * improvment extra and tooltip * Small change when using auto grey as start reference * Remove itcwb_fgreen from GUI * Various change - and 'unsatisfactory' test to take into account Custom * Change second temp when < 4000 - replace automatic by automatic and refinement * Neutralize 'Use custom tempertaure and green' * Green refinement * Clean code - small changes * Other clean code * Added forgotten itcw_green in procparams.cc * Fixed a bug dispaly deltaE * Change threshold settings - change code for Custom * various change in case of using camera settings * Change way to estimate range green * clean code and message in console * Clean and optimize * Disable 2 passes if custom * Improve GUI 2 passes * Increase sampling and number of temperature above 9400K * Hide AWB temperature bais and green refinement when Custom temerature ans green * marks out the code with past time markers * Improve time processing with spectral colors temp calculations * Replace in console msec by nsec * Scan temp 5.9 * Change max temp and behavior if temp > 15000k * Refine temp near 14000K * Increase sampling temperature 132 to 167 * Increase sampling temperature * Set in preference choice for itcwb_custom * White point calculated with observer instead of table * More on whitepoint calculation - informations * Clean and format code with astyle colortemp.cc rawimagesource.cc improccoordinator.cc * Small change tooltip * Improve update auto-wb * Remove 'use custom temperature and tin' - simplify sampling to 2 choice * Small change to rand AWB bias - green refinement - tooltips * slightly modify the parameters upstream of the algo - do not use camera in some cases * Mixed initial green * Refine init green and temp when no camera * Refine temp and green in do not use camera settings * Change tint threshold and various improvments (tooltips, wrong values..) * Refinment and clean code * Improve mode No use camera settings * Init nocam * Refine use no camera settings * Small change * 5.9 compatibility - and format * Refine sampling temperature around 7000 - 8000K * Format astylert rawimagesource colortemp * 5.9 settings * 5.9 settings 2 * Various change - 5.9 - format * Improve pre and post algorithm * Fixed correction temp > 8000K * Fixed bad behavior Tppat.minchroma * Change sampling choices and tooltip * Change name wp wp2 wip to wb wb2 iwb to avoid misunderstood * Small Change settings - limit display studgood to 0.0001 * Improvment display GUI temp * Change default sampling * Change ACESP0 to JDCmax * Improve GUI abstract profile * Allows dispaly in console xy values thanks to Reffort * Remove bruce rgb in wp and abstract * Change wp abstract and save itcwb with dispaly cellxy * Change prim = 13 improccoordinator iplab2rgb * remove display datas in console - fixed wrong display Y xyY * Fixed ifferences in GUI maxhist minhist * Change limits x y in function sampling * fixed a dizziness * Change after algorithm rule when temp > 6000K * Harmonize limx limy for oldsampling 5.9 * Added Cam xyz matrix to sampling - thanks to Reffort * Display in console xyz matrix * Forgotten instruction when mix camera and autowb grey * Change rgbloc with Camera matrix Dcraw when no input profile * Change rep to repref in rgbxyz * Remove message in console * 5.9 compat * 5.9 compat 2 * 5.9 compat 3 * 5.9 compat 4 * 5.9 compat 5 format * 5.9 compta 6 * Improve code * Diasble itcwb_sampling in GUI * Comment code * Clean and comment code * 5 spectral colors * 9 spectral colors * 2 spectral colors * Various improvments * 5 spectral colors * 5 spectral colors * First changes review lawrence37 * Second changes review Lawrence37 * 3rd changes review Lawrence37 * 4th changes review Lawrence37 * Change windows and appimage yml * Re-enable Rtv4_Bruce in procparams.cc * test for 5.9 compatibility * Try to fixed crash * Slightly improve the management of 5.9 * test for 5.9 compatibility by passing parameters via the GUI * Forgotten parameter * Change ref_spec limit and remove pac_start for itcwb_sampling * Puts back the old Planck formula - the same as in dev * Revert "test for 5.9 compatibility by passing parameters via the GUI" This reverts commit d6b5a8feda31b93a5574cee6ebf4d3e9d58bbcfb. * Switch to new ITCWB only when changing WB method Prevents sudden changing of the white balance if ITCWB is selected and the image is reopened. * Fix spelling in comment * Suggested change from Lawrence37 to whitebalance - small change to improcoordinator * tempbias Set_sentive(false) if itcwb_smapling - clean whitebalance * Another clean whitebalance.cc --------- Co-authored-by: Lawrence Lee <45837045+Lawrence37@users.noreply.github.com>
387 lines
11 KiB
C++
387 lines
11 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/>.
|
|
*/
|
|
#include "stdimagesource.h"
|
|
|
|
#include "color.h"
|
|
#include "iccstore.h"
|
|
#include "image8.h"
|
|
#include "image16.h"
|
|
#include "imagefloat.h"
|
|
#include "imageio.h"
|
|
#include "mytime.h"
|
|
#include "procparams.h"
|
|
#include "utils.h"
|
|
|
|
#undef THREAD_PRIORITY_NORMAL
|
|
|
|
namespace rtengine
|
|
{
|
|
using namespace procparams;
|
|
ProcParams* params;
|
|
|
|
template<class T> void freeArray (T** a, int H)
|
|
{
|
|
for (int i = 0; i < H; i++) {
|
|
delete [] a[i];
|
|
}
|
|
|
|
delete [] a;
|
|
}
|
|
template<class T> T** allocArray (int W, int H)
|
|
{
|
|
|
|
T** t = new T*[H];
|
|
|
|
for (int i = 0; i < H; i++) {
|
|
t[i] = new T[W];
|
|
}
|
|
|
|
return t;
|
|
}
|
|
|
|
StdImageSource::StdImageSource () : ImageSource(), img(nullptr), plistener(nullptr), full(false), max{}, rgbSourceModified(false)
|
|
{
|
|
|
|
embProfile = nullptr;
|
|
idata = nullptr;
|
|
}
|
|
|
|
StdImageSource::~StdImageSource ()
|
|
{
|
|
|
|
delete idata;
|
|
|
|
if (img) {
|
|
delete img;
|
|
}
|
|
}
|
|
|
|
void StdImageSource::getSampleFormat (const Glib::ustring &fname, IIOSampleFormat &sFormat, IIOSampleArrangement &sArrangement)
|
|
{
|
|
|
|
sFormat = IIOSF_UNKNOWN;
|
|
sArrangement = IIOSA_UNKNOWN;
|
|
|
|
if (hasJpegExtension(fname)) {
|
|
// For now, png and jpeg files are converted to unsigned short by the loader itself,
|
|
// but there should be functions that read the sample format first, like the TIFF case below
|
|
sFormat = IIOSF_UNSIGNED_CHAR;
|
|
sArrangement = IIOSA_CHUNKY;
|
|
return;
|
|
} else if (hasPngExtension(fname)) {
|
|
int result = ImageIO::getPNGSampleFormat (fname, sFormat, sArrangement);
|
|
|
|
if (result == IMIO_SUCCESS) {
|
|
return;
|
|
}
|
|
} else if (hasTiffExtension(fname)) {
|
|
int result = ImageIO::getTIFFSampleFormat (fname, sFormat, sArrangement);
|
|
|
|
if (result == IMIO_SUCCESS) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* This method make define the correspondence between the input image type
|
|
* and RT's image data type (Image8, Image16 and Imagefloat), then it will
|
|
* load the image into it
|
|
*/
|
|
int StdImageSource::load (const Glib::ustring &fname)
|
|
{
|
|
|
|
fileName = fname;
|
|
|
|
// First let's find out the input image's type
|
|
|
|
IIOSampleFormat sFormat;
|
|
IIOSampleArrangement sArrangement;
|
|
getSampleFormat(fname, sFormat, sArrangement);
|
|
|
|
// Then create the appropriate object
|
|
|
|
switch (sFormat) {
|
|
case (IIOSF_UNSIGNED_CHAR): {
|
|
img = new Image8;
|
|
break;
|
|
}
|
|
|
|
case (IIOSF_UNSIGNED_SHORT): {
|
|
img = new Image16;
|
|
break;
|
|
}
|
|
|
|
case (IIOSF_LOGLUV24):
|
|
case (IIOSF_LOGLUV32):
|
|
case (IIOSF_FLOAT16):
|
|
case (IIOSF_FLOAT24):
|
|
case (IIOSF_FLOAT32): {
|
|
img = new Imagefloat;
|
|
break;
|
|
}
|
|
|
|
default:
|
|
return IMIO_FILETYPENOTSUPPORTED;
|
|
}
|
|
|
|
img->setSampleFormat(sFormat);
|
|
img->setSampleArrangement(sArrangement);
|
|
|
|
if (plistener) {
|
|
plistener->setProgressStr ("PROGRESSBAR_LOADING");
|
|
plistener->setProgress (0.0);
|
|
img->setProgressListener (plistener);
|
|
}
|
|
|
|
// And load the image!
|
|
|
|
int error = img->load (fname);
|
|
|
|
if (error) {
|
|
delete img;
|
|
img = nullptr;
|
|
return error;
|
|
}
|
|
|
|
embProfile = img->getEmbeddedProfile ();
|
|
|
|
idata = new FramesData (fname);
|
|
|
|
if (idata->hasExif()) {
|
|
int deg = 0;
|
|
|
|
if (idata->getOrientation() == "Rotate 90 CW") {
|
|
deg = 90;
|
|
} else if (idata->getOrientation() == "Rotate 180") {
|
|
deg = 180;
|
|
} else if (idata->getOrientation() == "Rotate 270 CW") {
|
|
deg = 270;
|
|
}
|
|
|
|
if (deg) {
|
|
img->rotate(deg);
|
|
}
|
|
}
|
|
|
|
if (plistener) {
|
|
plistener->setProgressStr ("PROGRESSBAR_READY");
|
|
plistener->setProgress (1.0);
|
|
}
|
|
|
|
wb = ColorTemp (1.0, 1.0, 1.0, 1.0, ColorTemp::DEFAULT_OBSERVER);
|
|
//this is probably a mistake if embedded profile is not D65
|
|
|
|
return 0;
|
|
}
|
|
|
|
void StdImageSource::getImage (const ColorTemp &ctemp, int tran, Imagefloat* image, const PreviewProps &pp, const ToneCurveParams &hrp, const RAWParams &raw, int opposed)
|
|
{
|
|
|
|
// the code will use OpenMP as of now.
|
|
|
|
img->getStdImage(ctemp, tran, image, pp);
|
|
|
|
// Hombre: we could have rotated the image here too, with just few line of code, but:
|
|
// 1. it would require other modifications in the engine, so "do not touch that little plonker!"
|
|
// 2. it's more optimized like this
|
|
|
|
// Flip if needed
|
|
if (tran & TR_HFLIP) {
|
|
image->hflip();
|
|
}
|
|
|
|
if (tran & TR_VFLIP) {
|
|
image->vflip();
|
|
}
|
|
}
|
|
|
|
void StdImageSource::convertColorSpace(Imagefloat* image, const ColorManagementParams &cmp, const ColorTemp &wb)
|
|
{
|
|
colorSpaceConversion (image, cmp, embProfile, img->getSampleFormat());
|
|
}
|
|
|
|
void StdImageSource::colorSpaceConversion (Imagefloat* im, const ColorManagementParams &cmp, cmsHPROFILE embedded, IIOSampleFormat sampleFormat)
|
|
{
|
|
|
|
bool skipTransform = false;
|
|
cmsHPROFILE in = nullptr;
|
|
cmsHPROFILE out = ICCStore::getInstance()->workingSpace (cmp.workingProfile);
|
|
|
|
if (cmp.inputProfile == "(embedded)" || cmp.inputProfile.empty() || cmp.inputProfile == "(camera)" || cmp.inputProfile == "(cameraICC)") {
|
|
if (embedded) {
|
|
in = embedded;
|
|
} else {
|
|
if (sampleFormat & (IIOSF_LOGLUV24 | IIOSF_LOGLUV32 | IIOSF_FLOAT16 | IIOSF_FLOAT24 | IIOSF_FLOAT32)) {
|
|
skipTransform = true;
|
|
} else {
|
|
in = ICCStore::getInstance()->getsRGBProfile ();
|
|
}
|
|
}
|
|
} else {
|
|
if (cmp.inputProfile != "(none)") {
|
|
in = ICCStore::getInstance()->getProfile (cmp.inputProfile);
|
|
|
|
if (in == nullptr && embedded) {
|
|
in = embedded;
|
|
} else if (in == nullptr) {
|
|
if (sampleFormat & (IIOSF_LOGLUV24 | IIOSF_LOGLUV32 | IIOSF_FLOAT16 | IIOSF_FLOAT24 | IIOSF_FLOAT32)) {
|
|
skipTransform = true;
|
|
} else {
|
|
in = ICCStore::getInstance()->getsRGBProfile ();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!skipTransform && in) {
|
|
if(in == embedded && cmsGetColorSpace(in) != cmsSigRgbData) { // if embedded profile is not an RGB profile, use sRGB
|
|
printf("embedded profile is not an RGB profile, using sRGB as input profile\n");
|
|
in = ICCStore::getInstance()->getsRGBProfile ();
|
|
}
|
|
|
|
lcmsMutex->lock ();
|
|
cmsHTRANSFORM hTransform = cmsCreateTransform (in, TYPE_RGB_FLT, out, TYPE_RGB_FLT, INTENT_RELATIVE_COLORIMETRIC,
|
|
cmsFLAGS_NOOPTIMIZE | cmsFLAGS_NOCACHE);
|
|
lcmsMutex->unlock ();
|
|
|
|
if(hTransform) {
|
|
// Convert to the [0.0 ; 1.0] range
|
|
im->normalizeFloatTo1();
|
|
|
|
im->ExecCMSTransform(hTransform);
|
|
|
|
// Converting back to the [0.0 ; 65535.0] range
|
|
im->normalizeFloatTo65535();
|
|
|
|
cmsDeleteTransform(hTransform);
|
|
} else {
|
|
printf("Could not convert from %s to %s\n", in == embedded ? "embedded profile" : cmp.inputProfile.data(), cmp.workingProfile.data());
|
|
}
|
|
}
|
|
}
|
|
|
|
void StdImageSource::getFullSize (int& w, int& h, int tr)
|
|
{
|
|
|
|
w = img->getWidth();
|
|
h = img->getHeight();
|
|
|
|
if ((tr & TR_ROT) == TR_R90 || (tr & TR_ROT) == TR_R270) {
|
|
w = img->getHeight();
|
|
h = img->getWidth();
|
|
}
|
|
}
|
|
|
|
void StdImageSource::getSize (const PreviewProps &pp, int& w, int& h)
|
|
{
|
|
w = pp.getWidth() / pp.getSkip() + (pp.getWidth() % pp.getSkip() > 0);
|
|
h = pp.getHeight() / pp.getSkip() + (pp.getHeight() % pp.getSkip() > 0);
|
|
}
|
|
|
|
void StdImageSource::getAutoExpHistogram (LUTu & histogram, int& histcompr)
|
|
{
|
|
if (img->getType() == sImage8) {
|
|
Image8 *img_ = static_cast<Image8*>(img);
|
|
img_->computeAutoHistogram(histogram, histcompr);
|
|
} else if (img->getType() == sImage16) {
|
|
Image16 *img_ = static_cast<Image16*>(img);
|
|
img_->computeAutoHistogram(histogram, histcompr);
|
|
} else if (img->getType() == sImagefloat) {
|
|
Imagefloat *img_ = static_cast<Imagefloat*>(img);
|
|
img_->computeAutoHistogram(histogram, histcompr);
|
|
}
|
|
}
|
|
|
|
void StdImageSource::WBauto(bool extra, 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 &temp0, float &delta, int &bia, int &dread, int &kcam, int &nocam, float &studgood, float &minchrom, int &kmin, float &minhist, float &maxhist, bool &twotimes, const WBParams & wbpar, int begx, int begy, int yEn, int xEn, int cx, int cy, const ColorManagementParams &cmp, const RAWParams &raw, const ToneCurveParams &hrp)
|
|
{
|
|
}
|
|
|
|
void StdImageSource::getAutoWBMultipliersitc(bool extra, double &tempref, double &greenref, double &tempitc, double &greenitc, float &temp0, float &delta, int &bia, int &dread, int &kcam, int &nocam, float &studgood, float &minchrom, int &kmin, float &minhist, float &maxhist, 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 WBParams & wbpar, const ColorManagementParams &cmp, const RAWParams &raw, const ToneCurveParams &hrp)
|
|
{
|
|
if (redAWBMul != -1.) {
|
|
rm = redAWBMul;
|
|
gm = greenAWBMul;
|
|
bm = blueAWBMul;
|
|
return;
|
|
}
|
|
|
|
img->getAutoWBMultipliersitc(extra, tempref, greenref, tempitc, greenitc, temp0, delta, bia, dread, kcam, nocam, studgood, minchrom, kmin, minhist, maxhist, begx, begy, yEn, xEn, cx, cy, bf_h, bf_w, rm, gm, bm, params->wb, params->icm, params->raw, params->toneCurve);
|
|
|
|
redAWBMul = rm;
|
|
greenAWBMul = gm;
|
|
blueAWBMul = bm;
|
|
}
|
|
|
|
|
|
void StdImageSource::getAutoWBMultipliers (double &rm, double &gm, double &bm)
|
|
{
|
|
if (redAWBMul != -1.) {
|
|
rm = redAWBMul;
|
|
gm = greenAWBMul;
|
|
bm = blueAWBMul;
|
|
return;
|
|
}
|
|
|
|
img->getAutoWBMultipliers(rm, gm, bm);
|
|
|
|
redAWBMul = rm;
|
|
greenAWBMul = gm;
|
|
blueAWBMul = bm;
|
|
}
|
|
|
|
ColorTemp StdImageSource::getSpotWB (std::vector<Coord2D> &red, std::vector<Coord2D> &green, std::vector<Coord2D>& blue, int tran, double equal, StandardObserver observer)
|
|
{
|
|
int rn, gn, bn;
|
|
double reds, greens, blues;
|
|
img->getSpotWBData(reds, greens, blues, rn, gn, bn, red, green, blue, tran);
|
|
double img_r, img_g, img_b;
|
|
wb.getMultipliers (img_r, img_g, img_b);
|
|
|
|
if( settings->verbose ) {
|
|
printf ("AVG: %g %g %g\n", reds / rn, greens / gn, blues / bn);
|
|
}
|
|
|
|
return ColorTemp (reds / rn * img_r, greens / gn * img_g, blues / bn * img_b, equal, observer);
|
|
}
|
|
|
|
void StdImageSource::flush() {
|
|
img->allocate(0, 0);
|
|
};
|
|
|
|
void StdImageSource::wbMul2Camera(double &rm, double &gm, double &bm)
|
|
{
|
|
rm = 1.0 / rm;
|
|
gm = 1.0 / gm;
|
|
bm = 1.0 / bm;
|
|
}
|
|
|
|
|
|
void StdImageSource::wbCamera2Mul(double &rm, double &gm, double &bm)
|
|
{
|
|
rm = 1.0 / rm;
|
|
gm = 1.0 / gm;
|
|
bm = 1.0 / bm;
|
|
}
|
|
|
|
}
|
|
|