/* -*- C++ -*- * * This file is part of RawTherapee. * * Copyright 2018 Alberto Griggio * * 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 . */ #ifdef _OPENMP #include #endif #include "improcfun.h" namespace rtengine { namespace { inline float sl(float blend, float x) { if (!OOG(x)) { const float orig = 1.f - blend; float v = Color::gamma_srgb(x) / MAXVALF; // Pegtop's formula from // https://en.wikipedia.org/wiki/Blend_modes#Soft_Light float v2 = v * v; float v22 = v2 * 2.f; v = v2 + v22 - v22 * v; x = blend * Color::igamma_srgb(v * MAXVALF) + orig * x; } return x; } } // namespace void ImProcFunctions::softLight(LabImage *lab) { if (!params->softlight.enabled || !params->softlight.strength) { return; } Imagefloat working(lab->W, lab->H); lab2rgb(*lab, working, params->icm.workingProfile); const float blend = params->softlight.strength / 100.f; #ifdef _OPENMP #pragma omp parallel for #endif for (int y = 0; y < working.getHeight(); ++y) { for (int x = 0; x < working.getWidth(); ++x) { working.r(y, x) = sl(blend, working.r(y, x)); working.g(y, x) = sl(blend, working.g(y, x)); working.b(y, x) = sl(blend, working.b(y, x)); } } rgb2lab(working, *lab, params->icm.workingProfile); } void ImProcFunctions::softLightloc(LabImage *lab, LabImage *dst, float blend) { // if (!params->softlight.enabled || !params->softlight.strength) { // return; // } Imagefloat working(lab->W, lab->H); lab2rgb(*lab, working, params->icm.workingProfile); // const float blend = stre / 100.f; #ifdef _OPENMP #pragma omp parallel for #endif for (int y = 0; y < working.getHeight(); ++y) { for (int x = 0; x < working.getWidth(); ++x) { working.r(y, x) = sl(blend, working.r(y, x)); working.g(y, x) = sl(blend, working.g(y, x)); working.b(y, x) = sl(blend, working.b(y, x)); } } rgb2lab(working, *dst, params->icm.workingProfile); } } // namespace rtengine