353 lines
11 KiB
C++
353 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/>.
|
|
*/
|
|
#pragma once
|
|
|
|
#include <gtkmm.h>
|
|
|
|
#include <glibmm/ustring.h>
|
|
|
|
#include <cairomm/cairomm.h>
|
|
|
|
#include "guiutils.h"
|
|
#include "pointermotionlistener.h"
|
|
|
|
#include "../rtengine/LUT.h"
|
|
#include "../rtengine/noncopyable.h"
|
|
|
|
class HistogramArea;
|
|
|
|
namespace
|
|
{
|
|
constexpr int VECTORSCOPE_SIZE = 128;
|
|
}
|
|
|
|
struct HistogramAreaIdleHelper {
|
|
HistogramArea* harea;
|
|
bool destroyed;
|
|
int pending;
|
|
};
|
|
|
|
class HistogramRGBArea;
|
|
struct HistogramRGBAreaIdleHelper {
|
|
HistogramRGBArea* harea;
|
|
bool destroyed;
|
|
int pending;
|
|
};
|
|
|
|
class HistogramScaling
|
|
{
|
|
public:
|
|
double factor;
|
|
HistogramScaling() : factor(10.0) {}
|
|
double log (double vsize, double val);
|
|
};
|
|
|
|
class HistogramRGBArea : public Gtk::DrawingArea, public BackBuffer, protected HistogramScaling, public rtengine::NonCopyable
|
|
{
|
|
private:
|
|
typedef const double (*TMatrix)[3];
|
|
|
|
IdleRegister idle_register;
|
|
|
|
protected:
|
|
int val;
|
|
int r;
|
|
int g;
|
|
int b;
|
|
|
|
bool valid;
|
|
|
|
bool needRed;
|
|
bool needGreen;
|
|
bool needBlue;
|
|
bool needLuma;
|
|
bool needChroma;
|
|
bool rawMode;
|
|
bool showMode;
|
|
bool barDisplayed;
|
|
|
|
Gtk::Grid* parent;
|
|
|
|
double padding = 5.0;
|
|
|
|
HistogramRGBAreaIdleHelper* harih;
|
|
|
|
/** Draw an indicator bar for the value. */
|
|
virtual void drawBar(Cairo::RefPtr<Cairo::Context> cc, double value, double max_value, int winw, int winh, double scale) = 0;
|
|
|
|
void getPreferredThickness(int& min_thickness, int& natural_length) const;
|
|
void getPreferredLength(int& min_length, int& natural_length) const;
|
|
void getPreferredThicknessForLength(int length, int& min_thickness, int& natural_length) const;
|
|
void getPreferredLengthForThickness(int thickness, int& min_length, int& natural_length) const;
|
|
|
|
public:
|
|
HistogramRGBArea();
|
|
~HistogramRGBArea() override;
|
|
|
|
void updateBackBuffer (int r, int g, int b, const Glib::ustring &profile = "", const Glib::ustring &profileW = "");
|
|
bool getShow ();
|
|
void setShow(bool show);
|
|
void setParent (Gtk::Grid* p)
|
|
{
|
|
parent = p;
|
|
};
|
|
|
|
void update (int val, int rh, int gh, int bh);
|
|
void updateOptions (bool r, bool g, bool b, bool l, bool c, bool raw, bool show);
|
|
|
|
void on_realize() override;
|
|
bool on_draw(const ::Cairo::RefPtr< Cairo::Context> &cr) override;
|
|
bool on_button_press_event (GdkEventButton* event) override;
|
|
void factorChanged (double newFactor);
|
|
|
|
};
|
|
|
|
class HistogramRGBAreaHori final : public HistogramRGBArea
|
|
{
|
|
private:
|
|
void drawBar(Cairo::RefPtr<Cairo::Context> cc, double value, double max_value, int winw, int winh, double scale) override;
|
|
|
|
Gtk::SizeRequestMode get_request_mode_vfunc () const override;
|
|
void get_preferred_height_vfunc (int& minimum_height, int& natural_height) const override;
|
|
void get_preferred_width_vfunc (int &minimum_width, int &natural_width) const override;
|
|
void get_preferred_height_for_width_vfunc (int width, int &minimum_height, int &natural_height) const override;
|
|
void get_preferred_width_for_height_vfunc (int h, int &minimum_width, int &natural_width) const override;
|
|
};
|
|
|
|
class HistogramRGBAreaVert final : public HistogramRGBArea
|
|
{
|
|
private:
|
|
void drawBar(Cairo::RefPtr<Cairo::Context> cc, double value, double max_value, int winw, int winh, double scale) override;
|
|
|
|
Gtk::SizeRequestMode get_request_mode_vfunc () const override;
|
|
void get_preferred_height_vfunc (int& minimum_height, int& natural_height) const override;
|
|
void get_preferred_width_vfunc (int &minimum_width, int &natural_width) const override;
|
|
void get_preferred_height_for_width_vfunc (int width, int &minimum_height, int &natural_height) const override;
|
|
void get_preferred_width_for_height_vfunc (int h, int &minimum_width, int &natural_width) const override;
|
|
};
|
|
|
|
class DrawModeListener
|
|
{
|
|
public:
|
|
virtual ~DrawModeListener() = default;
|
|
virtual void toggleButtonMode() = 0;
|
|
};
|
|
|
|
class HistogramArea final : public Gtk::DrawingArea, public BackBuffer, private HistogramScaling, public rtengine::NonCopyable
|
|
{
|
|
public:
|
|
typedef sigc::signal<void, double> type_signal_factor_changed;
|
|
|
|
private:
|
|
IdleRegister idle_register;
|
|
type_signal_factor_changed sigFactorChanged;
|
|
|
|
protected:
|
|
LUTu rhist, ghist, bhist, lhist, chist;
|
|
LUTu rhistRaw, ghistRaw, bhistRaw, lhistRaw; //lhistRaw is unused?
|
|
int vectorscope_scale;
|
|
int vect[VECTORSCOPE_SIZE][VECTORSCOPE_SIZE];
|
|
std::unique_ptr<unsigned char[]> vect_buffer;
|
|
bool vect_buffer_dirty;
|
|
int waveform_scale;
|
|
int waveform_width;
|
|
std::unique_ptr<int[][256]> rwave, gwave, bwave, lwave;
|
|
std::unique_ptr<unsigned char[]> wave_buffer;
|
|
std::unique_ptr<unsigned char[]> wave_buffer_luma;
|
|
bool wave_buffer_dirty;
|
|
|
|
bool valid;
|
|
int drawMode;
|
|
DrawModeListener *myDrawModeListener;
|
|
int scopeType;
|
|
int oldwidth, oldheight;
|
|
/// Intensity of waveform and vectorscope trace.
|
|
float trace_brightness;
|
|
|
|
bool needRed, needGreen, needBlue, needLuma, needChroma;
|
|
bool rawMode;
|
|
bool isPressed;
|
|
double movingPosition;
|
|
bool needPointer;
|
|
|
|
double padding = 5.0;
|
|
|
|
HistogramAreaIdleHelper* haih;
|
|
|
|
int pointer_red, pointer_green, pointer_blue;
|
|
float pointer_a, pointer_b;
|
|
|
|
public:
|
|
explicit HistogramArea(DrawModeListener *fml = nullptr);
|
|
~HistogramArea() override;
|
|
|
|
void updateBackBuffer ();
|
|
/// Update pointer values. Returns true if widget needs redrawing.
|
|
bool updatePointer(int r, int g, int b, const Glib::ustring &profile = "", const Glib::ustring &profileW = "");
|
|
void update(
|
|
const LUTu& histRed,
|
|
const LUTu& histGreen,
|
|
const LUTu& histBlue,
|
|
const LUTu& histLuma,
|
|
const LUTu& histChroma,
|
|
const LUTu& histRedRaw,
|
|
const LUTu& histGreenRaw,
|
|
const LUTu& histBlueRaw,
|
|
int vectorscopeScale,
|
|
const int vectorscope[VECTORSCOPE_SIZE][VECTORSCOPE_SIZE],
|
|
int waveformScale,
|
|
int waveformWidth,
|
|
const int waveformRed[][256],
|
|
const int waveformGreen[][256],
|
|
const int waveformBlue[][256],
|
|
const int waveformLuma[][256]
|
|
);
|
|
void updateOptions (bool r, bool g, bool b, bool l, bool c, bool raw, int mode, int type, bool pointer);
|
|
void on_realize() override;
|
|
bool on_draw(const ::Cairo::RefPtr< Cairo::Context> &cr) override;
|
|
bool on_button_press_event (GdkEventButton* event) override;
|
|
bool on_button_release_event (GdkEventButton* event) override;
|
|
bool on_motion_notify_event (GdkEventMotion* event) override;
|
|
type_signal_factor_changed signal_factor_changed();
|
|
|
|
private:
|
|
void drawCurve(Cairo::RefPtr<Cairo::Context> &cr, const LUTu & data, double scale, int hsize, int vsize);
|
|
void drawMarks(Cairo::RefPtr<Cairo::Context> &cr, const LUTu & data, double scale, int hsize, int & ui, int & oi);
|
|
void drawVectorscope(Cairo::RefPtr<Cairo::Context> &cr, int hsize, int vsize);
|
|
void drawWaveform(Cairo::RefPtr<Cairo::Context> &cr, int hsize, int vsize);
|
|
Gtk::SizeRequestMode get_request_mode_vfunc () const override;
|
|
void get_preferred_height_vfunc (int& minimum_height, int& natural_height) const override;
|
|
void get_preferred_width_vfunc (int &minimum_width, int &natural_width) const override;
|
|
void get_preferred_height_for_width_vfunc (int width, int &minimum_height, int &natural_height) const override;
|
|
void get_preferred_width_for_height_vfunc (int height, int &minimum_width, int &natural_width) const override;
|
|
};
|
|
|
|
class HistogramPanelListener
|
|
{
|
|
public:
|
|
enum ScopeType {HISTOGRAM, VECTORSCOPE_CH, VECTORSCOPE_HS, WAVEFORM, NONE};
|
|
|
|
virtual void scopeTypeChanged(ScopeType new_type) = 0;
|
|
};
|
|
|
|
class HistogramPanel final : public Gtk::Grid, public PointerMotionListener, public DrawModeListener, public rtengine::NonCopyable
|
|
{
|
|
|
|
protected:
|
|
|
|
Gtk::Grid* gfxGrid;
|
|
Gtk::Grid* buttonGrid;
|
|
HistogramArea* histogramArea;
|
|
HistogramRGBArea* histogramRGBArea;
|
|
std::unique_ptr<HistogramRGBAreaHori> histogramRGBAreaHori;
|
|
std::unique_ptr<HistogramRGBAreaVert> histogramRGBAreaVert;
|
|
Gtk::ToggleButton* showRed;
|
|
Gtk::ToggleButton* showGreen;
|
|
Gtk::ToggleButton* showBlue;
|
|
Gtk::ToggleButton* showValue;
|
|
Gtk::ToggleButton* showRAW;
|
|
Gtk::ToggleButton* showBAR;
|
|
Gtk::ToggleButton* showChro;
|
|
Gtk::Button* showMode;
|
|
Gtk::Button* scopeType;
|
|
|
|
Gtk::Image *redImage;
|
|
Gtk::Image *greenImage;
|
|
Gtk::Image *blueImage;
|
|
Gtk::Image *valueImage;
|
|
Gtk::Image *rawImage;
|
|
Gtk::Image *barImage;
|
|
Gtk::Image *chroImage;
|
|
|
|
Gtk::Image *redImage_g;
|
|
Gtk::Image *greenImage_g;
|
|
Gtk::Image *blueImage_g;
|
|
Gtk::Image *valueImage_g;
|
|
Gtk::Image *rawImage_g;
|
|
Gtk::Image *barImage_g;
|
|
Gtk::Image *chroImage_g;
|
|
|
|
std::unique_ptr<Gtk::Image> histImage;
|
|
std::unique_ptr<Gtk::Image> waveImage;
|
|
std::unique_ptr<Gtk::Image> vectHcImage;
|
|
std::unique_ptr<Gtk::Image> vectHsImage;
|
|
|
|
Gtk::Image *mode0Image;
|
|
Gtk::Image *mode1Image;
|
|
Gtk::Image *mode2Image;
|
|
|
|
HistogramPanelListener* panel_listener;
|
|
|
|
sigc::connection rconn;
|
|
void setHistInvalid ();
|
|
void showRGBBar();
|
|
void updateHistAreaOptions();
|
|
|
|
public:
|
|
|
|
HistogramPanel ();
|
|
~HistogramPanel () override;
|
|
|
|
void histogramChanged(
|
|
const LUTu& histRed,
|
|
const LUTu& histGreen,
|
|
const LUTu& histBlue,
|
|
const LUTu& histLuma,
|
|
const LUTu& histChroma,
|
|
const LUTu& histRedRaw,
|
|
const LUTu& histGreenRaw,
|
|
const LUTu& histBlueRaw,
|
|
int vectorscopeScale,
|
|
const int vectorscope[VECTORSCOPE_SIZE][VECTORSCOPE_SIZE],
|
|
int waveformScale,
|
|
int waveformWidth,
|
|
const int waveformRed[][256],
|
|
const int waveformGreen[][256],
|
|
const int waveformBlue[][256],
|
|
const int waveformLuma[][256]
|
|
)
|
|
{
|
|
histogramArea->update(histRed, histGreen, histBlue, histLuma, histChroma, histRedRaw, histGreenRaw, histBlueRaw, vectorscopeScale, vectorscope, waveformScale, waveformWidth, waveformRed, waveformGreen, waveformBlue, waveformLuma);
|
|
}
|
|
// pointermotionlistener interface
|
|
void pointerMoved (bool validPos, const Glib::ustring &profile, const Glib::ustring &profileW, int x, int y, int r, int g, int b, bool isRaw = false) override;
|
|
|
|
// TODO should be protected
|
|
void setHistRGBInvalid ();
|
|
|
|
void reorder (Gtk::PositionType position);
|
|
void red_toggled ();
|
|
void green_toggled ();
|
|
void blue_toggled ();
|
|
void value_toggled ();
|
|
void raw_toggled ();
|
|
void chro_toggled ();
|
|
void bar_toggled ();
|
|
void mode_released ();
|
|
void type_pressed ();
|
|
void type_changed ();
|
|
void rgbv_toggled ();
|
|
void resized (Gtk::Allocation& req);
|
|
|
|
// drawModeListener interface
|
|
void toggleButtonMode () override;
|
|
|
|
void setPanelListener(HistogramPanelListener* listener);
|
|
};
|