rawTherapee/rtgui/imagearea.cc

769 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 <http://www.gnu.org/licenses/>.
*/
#include "imagearea.h"
#include <ctime>
#include <cmath>
#include "options.h"
#include "multilangmgr.h"
#include "cropwindow.h"
#include "../rtengine/refreshmap.h"
#include "options.h"
ImageArea::ImageArea (ImageAreaPanel* p) : parent(p), fullImageWidth(0), fullImageHeight(0)
{
infotext = "";
cropgl = nullptr;
pmlistener = nullptr;
pmhlistener = nullptr;
focusGrabber = nullptr;
flawnOverWindow = nullptr;
mainCropWindow = nullptr;
previewHandler = nullptr;
showClippedH = false;
showClippedS = false;
listener = nullptr;
zoomPanel = Gtk::manage (new ZoomPanel (this));
indClippedPanel = Gtk::manage (new IndicateClippedPanel (this));
previewModePanel = Gtk::manage (new PreviewModePanel (this));
add_events(Gdk::LEAVE_NOTIFY_MASK);
signal_size_allocate().connect( sigc::mem_fun(*this, &ImageArea::on_resized) );
dirty = false;
ipc = nullptr;
iLinkedImageArea = nullptr;
}
ImageArea::~ImageArea ()
{
for (auto cropWin : cropWins) {
delete cropWin;
}
cropWins.clear ();
if (mainCropWindow) {
delete mainCropWindow;
}
}
void ImageArea::on_realize()
{
Gtk::DrawingArea::on_realize();
#if defined (__APPLE__)
// Workaround: disabling POINTER_MOTION_HINT_MASK as for gtk 2.24.22 the get_pointer() function is buggy for quartz and modifier mask is not updated correctly.
// This workaround should be removed when bug is fixed in GTK2 or when migrating to GTK3
add_events(Gdk::EXPOSURE_MASK | Gdk::POINTER_MOTION_MASK | Gdk::BUTTON_PRESS_MASK | Gdk::BUTTON_RELEASE_MASK | Gdk::SCROLL_MASK);
#else
add_events(Gdk::EXPOSURE_MASK | Gdk::POINTER_MOTION_MASK | Gdk::POINTER_MOTION_HINT_MASK | Gdk::BUTTON_PRESS_MASK | Gdk::BUTTON_RELEASE_MASK | Gdk::SCROLL_MASK);
#endif
Cairo::FontOptions cfo;
cfo.set_antialias (Cairo::ANTIALIAS_SUBPIXEL);
get_pango_context ()->set_cairo_font_options (cfo);
}
void ImageArea::on_resized (Gtk::Allocation& req)
{
if (ipc && get_width() > 1) { // sometimes on_resize is called in some init state, causing wrong sizes
if (!mainCropWindow) {
mainCropWindow = new CropWindow (this, false, false);
mainCropWindow->setDecorated (false);
mainCropWindow->setFitZoomEnabled (true);
mainCropWindow->addCropWindowListener (this);
mainCropWindow->setCropGUIListener (cropgl);
mainCropWindow->setPointerMotionListener (pmlistener);
mainCropWindow->setPointerMotionHListener (pmhlistener);
mainCropWindow->setPosition (0, 0);
mainCropWindow->setSize (get_width(), get_height()); // this execute the refresh itself
mainCropWindow->enable(); // start processing !
} else {
mainCropWindow->setSize (get_width(), get_height()); // this execute the refresh itself
}
parent->syncBeforeAfterViews();
}
}
rtengine::StagedImageProcessor* ImageArea::getImProcCoordinator() const
{
return ipc;
}
void ImageArea::setImProcCoordinator(rtengine::StagedImageProcessor* ipc_)
{
if( !ipc_ ) {
focusGrabber = nullptr;
for (auto cropWin : cropWins) {
delete cropWin;
}
cropWins.clear();
mainCropWindow->deleteColorPickers ();
mainCropWindow->setObservedCropWin (nullptr);
}
ipc = ipc_;
}
void ImageArea::setPreviewHandler (PreviewHandler* ph)
{
previewHandler = ph;
}
void ImageArea::on_style_updated ()
{
// TODO: notify all crop windows that the style has been changed
queue_draw ();
}
void ImageArea::setInfoText (Glib::ustring text)
{
infotext = text;
Glib::RefPtr<Pango::Context> context = get_pango_context () ;
Pango::FontDescription fontd(get_style_context()->get_font());
// update font
fontd.set_weight (Pango::WEIGHT_BOLD);
fontd.set_size (10 * Pango::SCALE);
context->set_font_description (fontd);
// create text layout
Glib::RefPtr<Pango::Layout> ilayout = create_pango_layout("");
ilayout->set_markup(text);
// get size of the text block
int iw, ih;
ilayout->get_pixel_size (iw, ih);
// create BackBuffer
iBackBuffer.setDrawRectangle(Cairo::FORMAT_ARGB32, 0, 0, iw + 16, ih + 16, true);
iBackBuffer.setDestPosition(8, 8);
Cairo::RefPtr<Cairo::Context> cr = iBackBuffer.getContext();
// cleaning the back buffer (make it full transparent)
cr->set_source_rgba (0., 0., 0., 0.);
cr->set_operator (Cairo::OPERATOR_CLEAR);
cr->paint ();
cr->set_operator (Cairo::OPERATOR_OVER);
// paint transparent black background
cr->set_source_rgba (0., 0., 0., 0.5);
cr->paint ();
// paint text
cr->set_source_rgb (1.0, 1.0, 1.0);
cr->move_to (8, 8);
ilayout->add_to_cairo_context (cr);
cr->fill ();
}
void ImageArea::infoEnabled (bool e)
{
if (options.showInfo != e) {
options.showInfo = e;
queue_draw ();
}
}
CropWindow* ImageArea::getCropWindow (int x, int y)
{
CropWindow* cw = mainCropWindow;
for (auto cropWin : cropWins) {
if (cropWin->isInside (x, y)) {
return cropWin;
}
}
return cw;
}
void ImageArea::redraw ()
{
// dirty prevents multiple updates queued up
if (!dirty) {
dirty = true;
queue_draw ();
}
}
void ImageArea::switchPickerVisibility (bool isVisible)
{
redraw();
}
bool ImageArea::on_draw(const ::Cairo::RefPtr< Cairo::Context> &cr)
{
dirty = false;
/* HOMBRE: How do we replace that??
if (event->count) {
return true;
}
*/
if (mainCropWindow) {
mainCropWindow->expose (cr);
}
if (options.showInfo && infotext != "") {
iBackBuffer.copySurface(cr);
}
for (std::list<CropWindow*>::reverse_iterator i = cropWins.rbegin(); i != cropWins.rend(); ++i) {
(*i)->expose (cr);
}
return true;
}
bool ImageArea::on_motion_notify_event (GdkEventMotion* event)
{
if (focusGrabber) {
focusGrabber->pointerMoved (event->state, event->x, event->y);
} else {
CropWindow* cw = getCropWindow (event->x, event->y);
if (cw) {
if (cw != flawnOverWindow) {
if (flawnOverWindow) {
flawnOverWindow->flawnOver(false);
}
cw->flawnOver(true);
flawnOverWindow = cw;
}
cw->pointerMoved (event->state, event->x, event->y);
} else if (flawnOverWindow) {
flawnOverWindow->flawnOver(false);
flawnOverWindow = nullptr;
}
}
return true;
}
bool ImageArea::on_button_press_event (GdkEventButton* event)
{
if (focusGrabber) {
focusGrabber->buttonPress (event->button, event->type, event->state, event->x, event->y);
} else {
CropWindow* cw = getCropWindow (event->x, event->y);
if (cw) {
cw->buttonPress (event->button, event->type, event->state, event->x, event->y);
}
}
return true;
}
bool ImageArea::on_scroll_event (GdkEventScroll* event)
{
CropWindow* cw = getCropWindow (event->x, event->y);
if (cw) {
cw->scroll (event->state, event->direction, event->x, event->y);
}
return true;
}
bool ImageArea::on_button_release_event (GdkEventButton* event)
{
if (focusGrabber) {
focusGrabber->buttonRelease (event->button, event->type, event->state, event->x, event->y);
} else {
CropWindow* cw = getCropWindow (event->x, event->y);
if (cw) {
cw->buttonRelease (event->button, event->type, event->state, event->x, event->y);
}
}
return true;
}
bool ImageArea::on_leave_notify_event (GdkEventCrossing* event)
{
if (flawnOverWindow) {
flawnOverWindow->flawnOver(false);
flawnOverWindow = nullptr;
}
if (focusGrabber) {
focusGrabber->flawnOver(false);
focusGrabber->leaveNotify (event);
} else {
CropWindow* cw = getCropWindow (event->x, event->y);
if (cw) {
cw->flawnOver(false);
cw->leaveNotify (event);
}
}
return true;
}
void ImageArea::subscribe(EditSubscriber *subscriber)
{
EditDataProvider::subscribe(subscriber);
mainCropWindow->setEditSubscriber(subscriber);
for (auto cropWin : cropWins) {
cropWin->setEditSubscriber(subscriber);
}
if (listener && listener->getToolBar()) {
listener->getToolBar()->startEditMode ();
}
if (subscriber && subscriber->getEditingType() == ET_OBJECTS) {
// In this case, no need to reprocess the image, so we redraw the image to display the geometry
queue_draw();
}
}
void ImageArea::unsubscribe()
{
bool wasObjectType = false;
EditSubscriber* oldSubscriber = EditDataProvider::getCurrSubscriber();
if (oldSubscriber && oldSubscriber->getEditingType() == ET_OBJECTS) {
wasObjectType = true;
}
EditDataProvider::unsubscribe();
// Ask the Crops to free-up edit mode buffers
mainCropWindow->setEditSubscriber(nullptr);
for (auto cropWin : cropWins) {
cropWin->setEditSubscriber(nullptr);
}
setToolHand();
if (listener && listener->getToolBar()) {
listener->getToolBar()->stopEditMode ();
}
if (wasObjectType) {
queue_draw();
}
}
void ImageArea::getImageSize (int &w, int&h)
{
if (ipc) {
w = ipc->getFullWidth();
h = ipc->getFullHeight();
} else {
w = h = 0;
}
}
void ImageArea::grabFocus (CropWindow* cw)
{
focusGrabber = cw;
if (cw && cw != mainCropWindow) {
cropWindowSelected (cw);
}
}
void ImageArea::unGrabFocus ()
{
focusGrabber = nullptr;
}
void ImageArea::addCropWindow ()
{
if (!mainCropWindow) {
return; // if called but no image is loaded, it would crash
}
CropWindow* cw = new CropWindow (this, true, true);
cw->zoom11(false);
cw->setCropGUIListener (cropgl);
cw->setPointerMotionListener (pmlistener);
cw->setPointerMotionHListener (pmhlistener);
int lastWidth = options.detailWindowWidth;
int lastHeight = options.detailWindowHeight;
if (lastWidth < lastHeight) {
lastHeight = lastWidth;
}
if (lastHeight < lastWidth) {
lastWidth = lastHeight;
}
if(!cropWins.empty()) {
CropWindow *lastCrop;
lastCrop = cropWins.front();
if(lastCrop) {
lastCrop->getSize(lastWidth, lastHeight);
}
}
cropWins.push_front (cw);
// Position the new crop window this way: start from top right going down to bottom. When bottom is reached, continue top left going down......
int N = cropWins.size() - 1;
int cropwidth, cropheight;
if(lastWidth <= 0) { // this is only the case for very first start of RT 4.1 or when options file is deleted
cropwidth = 200;
cropheight = 200;
} else {
cropwidth = lastWidth;
cropheight = lastHeight;
}
cw->setSize (cropwidth, cropheight);
int x, y;
int maxRows = get_height() / cropheight;
if(maxRows == 0) {
maxRows = 1;
}
int col = N / maxRows;
if(col % 2) { // from left side
col = col / 2;
x = col * cropwidth;
if(x >= get_width() - 50) {
x = get_width() - 50;
}
} else { // from right side
col /= 2;
col++;
x = get_width() - col * cropwidth;
if(x <= 0) {
x = 0;
}
}
y = cropheight * (N % maxRows);
cw->setPosition (x, y);
cw->setEditSubscriber (getCurrSubscriber());
cw->enable(); // start processing!
{
int anchorX = 0;
int anchorY = 0;
mainCropWindow->getCropAnchorPosition(anchorX, anchorY);
cw->setCropAnchorPosition(anchorX, anchorY);
}
mainCropWindow->setObservedCropWin (cropWins.front());
if(!ipc->getHighQualComputed()) {
ipc->startProcessing(M_HIGHQUAL);
ipc->setHighQualComputed();
}
}
void ImageArea::cropWindowSelected (CropWindow* cw)
{
std::list<CropWindow*>::iterator i = std::find (cropWins.begin(), cropWins.end(), cw);
if (i != cropWins.end()) {
cropWins.erase (i);
}
cropWins.push_front (cw);
mainCropWindow->setObservedCropWin (cropWins.front());
}
void ImageArea::cropWindowClosed (CropWindow* cw)
{
focusGrabber = nullptr;
std::list<CropWindow*>::iterator i = std::find (cropWins.begin(), cropWins.end(), cw);
if (i != cropWins.end()) {
cropWins.erase (i);
}
if (!cropWins.empty()) {
mainCropWindow->setObservedCropWin (cropWins.front());
} else {
mainCropWindow->setObservedCropWin (nullptr);
}
queue_draw ();
}
void ImageArea::straightenReady (double rotDeg)
{
if (listener) {
listener->rotateSelectionReady (rotDeg);
}
}
void ImageArea::spotWBSelected (int x, int y)
{
if (listener) {
listener->spotWBselected (x, y);
}
}
void ImageArea::getScrollImageSize (int& w, int& h)
{
if (mainCropWindow && ipc) {
w = ipc->getFullWidth();
h = ipc->getFullHeight();
} else {
w = h = 0;
}
}
void ImageArea::getScrollPosition (int& x, int& y)
{
if (mainCropWindow) {
mainCropWindow->getCropAnchorPosition (x, y);
} else {
x = y = 0;
}
}
void ImageArea::setScrollPosition (int x, int y)
{
if (mainCropWindow) {
mainCropWindow->delCropWindowListener (this);
mainCropWindow->setCropAnchorPosition (x, y);
mainCropWindow->addCropWindowListener (this);
}
}
void ImageArea::cropPositionChanged (CropWindow* cw)
{
syncBeforeAfterViews ();
}
void ImageArea::cropWindowSizeChanged (CropWindow* cw)
{
syncBeforeAfterViews ();
}
void ImageArea::cropZoomChanged (CropWindow* cw)
{
if (cw == mainCropWindow) {
parent->zoomChanged ();
syncBeforeAfterViews ();
zoomPanel->refreshZoomLabel ();
}
}
double ImageArea::getZoom ()
{
if (mainCropWindow) {
return mainCropWindow->getZoom ();
} else {
return 1.0;
}
}
// Called by imageAreaPanel before/after views
void ImageArea::setZoom (double zoom)
{
if (mainCropWindow) {
mainCropWindow->setZoom (zoom);
}
zoomPanel->refreshZoomLabel ();
}
void ImageArea::initialImageArrived ()
{
if (mainCropWindow) {
int w, h;
mainCropWindow->cropHandler.getFullImageSize(w, h);
if(options.prevdemo != PD_Sidecar || !options.rememberZoomAndPan || w != fullImageWidth || h != fullImageHeight) {
if (options.cropAutoFit || options.bgcolor != 0) {
mainCropWindow->zoomFitCrop();
} else {
mainCropWindow->zoomFit();
}
} else if ((options.cropAutoFit || options.bgcolor != 0) && mainCropWindow->cropHandler.cropParams.enabled) {
mainCropWindow->zoomFitCrop();
}
fullImageWidth = w;
fullImageHeight = h;
}
}
void ImageArea::syncBeforeAfterViews ()
{
parent->syncBeforeAfterViews ();
}
void ImageArea::setCropGUIListener (CropGUIListener* l)
{
cropgl = l;
for (auto cropWin : cropWins) {
cropWin->setCropGUIListener (cropgl);
}
if (mainCropWindow) {
mainCropWindow->setCropGUIListener (cropgl);
}
}
void ImageArea::setPointerMotionListener (PointerMotionListener* pml)
{
pmlistener = pml;
for (auto cropWin : cropWins) {
cropWin->setPointerMotionListener (pml);
}
if (mainCropWindow) {
mainCropWindow->setPointerMotionListener (pml);
}
}
void ImageArea::setPointerMotionHListener (PointerMotionListener* pml)
{
pmhlistener = pml;
for (auto cropWin : cropWins) {
cropWin->setPointerMotionHListener (pml);
}
if (mainCropWindow) {
mainCropWindow->setPointerMotionHListener (pml);
}
}
ToolMode ImageArea::getToolMode ()
{
if (listener && listener->getToolBar()) {
return listener->getToolBar()->getTool ();
} else {
return TMHand;
}
}
bool ImageArea::showColorPickers ()
{
if (listener && listener->getToolBar()) {
return listener->getToolBar()->showColorPickers ();
} else {
return false;
}
}
void ImageArea::setToolHand ()
{
if (listener && listener->getToolBar()) {
listener->getToolBar()->setTool (TMHand);
}
}
int ImageArea::getSpotWBRectSize ()
{
if (listener) {
return listener->getSpotWBRectSize ();
} else {
return 1;
}
}
Gtk::SizeRequestMode ImageArea::get_request_mode_vfunc () const
{
return Gtk::SIZE_REQUEST_CONSTANT_SIZE;
}
void ImageArea::get_preferred_height_vfunc (int &minimum_height, int &natural_height) const
{
minimum_height= 50;
natural_height = 300;
}
void ImageArea::get_preferred_width_vfunc (int &minimum_width, int &natural_width) const
{
minimum_width = 100;
natural_width = 400;
}
void ImageArea::get_preferred_height_for_width_vfunc (int width, int &minimum_height, int &natural_height) const
{
get_preferred_height_vfunc(minimum_height, natural_height);
}
void ImageArea::get_preferred_width_for_height_vfunc (int height, int &minimum_width, int &natural_width) const
{
get_preferred_width_vfunc (minimum_width, natural_width);
}