]> git.sesse.net Git - kdenlive/blob - src/colorpickerwidget.cpp
0c971ef16a9948e62922c92e8e6c55cd5821e612
[kdenlive] / src / colorpickerwidget.cpp
1 /***************************************************************************
2  *   Copyright (C) 2010 by Till Theato (root@ttill.de)                     *
3  *                                                                         *
4  *   This program is free software; you can redistribute it and/or modify  *
5  *   it under the terms of the GNU General Public License as published by  *
6  *   the Free Software Foundation; either version 2 of the License, or     *
7  *   (at your option) any later version.                                   *
8  *                                                                         *
9  *   This program is distributed in the hope that it will be useful,       *
10  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        *
11  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         *
12  *   GNU General Public License for more details.                          *
13  *                                                                         *
14  *   You should have received a copy of the GNU General Public License     *
15  *   along with this program; if not, write to the                         *
16  *   Free Software Foundation, Inc.,                                       *
17  *   51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA          *
18  ***************************************************************************/
19
20
21 #include "colorpickerwidget.h"
22
23 #include <QMouseEvent>
24 #include <QHBoxLayout>
25 #include <QPushButton>
26 #include <QLabel>
27 #include <QSpinBox>
28 #include <QDesktopWidget>
29
30 #include <KApplication>
31 #include <KIcon>
32 #include <KLocalizedString>
33
34 #ifdef Q_WS_X11
35 #include <X11/Xutil.h>
36 #include <fixx11h.h>
37
38 class KCDPickerFilter: public QWidget
39 {
40 public:
41     KCDPickerFilter(QWidget* parent): QWidget(parent) {}
42
43     virtual bool x11Event(XEvent* event) {
44         if (event->type == ButtonRelease) {
45             QMouseEvent e(QEvent::MouseButtonRelease, QPoint(),
46             QPoint(event->xmotion.x_root, event->xmotion.y_root) , Qt::NoButton, Qt::NoButton, Qt::NoModifier);
47             QApplication::sendEvent(parentWidget(), &e);
48             return true;
49         }
50         return false;
51     }
52 };
53 #endif 
54
55
56 ColorPickerWidget::ColorPickerWidget(QWidget *parent) :
57         QWidget(parent),
58         m_filterActive(false)
59 {
60 #ifdef Q_WS_X11
61     m_filter = 0;
62     m_image = 0;
63 #endif
64
65     QHBoxLayout *layout = new QHBoxLayout(this);
66
67     QPushButton *button = new QPushButton(this);
68     button->setIcon(KIcon("color-picker"));
69     button->setToolTip(i18n("Pick a color on the screen"));
70     connect(button, SIGNAL(clicked()), this, SLOT(slotSetupEventFilter()));
71
72     m_size = new QSpinBox(this);
73     m_size->setMinimum(1);
74     // Use qMax here, to make it possible to get the average for the whole screen
75     m_size->setMaximum(qMax(qApp->desktop()->geometry().width(), qApp->desktop()->geometry().height()));
76     m_size->setValue(1);
77
78     layout->addWidget(button);
79     layout->addStretch(1);
80     layout->addWidget(new QLabel(i18n("Width of square to pick color from:")));
81     layout->addWidget(m_size);
82 }
83
84 ColorPickerWidget::~ColorPickerWidget()
85 {
86 #ifdef Q_WS_X11
87     if (m_filterActive && kapp)
88         kapp->removeX11EventFilter(m_filter);
89 #endif
90 }
91
92 QColor ColorPickerWidget::averagePickedColor(const QPoint pos)
93 {
94     int size = m_size->value();
95     int x0 = qMax(0, pos.x() - size / 2); 
96     int y0 = qMax(0, pos.y() - size / 2);
97     int x1 = qMin(qApp->desktop()->geometry().width(), pos.x() + size / 2);
98     int y1 = qMin(qApp->desktop()->geometry().height(), pos.y() + size / 2);
99
100     int sumR = 0;
101     int sumG = 0;
102     int sumB = 0;
103
104     /*
105      Only getting the XImage once for the whole rect,
106      results in a vast speed improvement.
107     */
108 #ifdef Q_WS_X11
109     Window root = RootWindow(QX11Info::display(), QX11Info::appScreen());
110     m_image = XGetImage(QX11Info::display(), root, x0, y0, x1 - x0, y1 - y0, -1, ZPixmap);
111 #endif
112
113     for (int i = x0; i < x1; ++i) {
114         for (int j = y0; j < y1; ++j) {
115             QColor color;
116 #ifdef Q_WS_X11
117             color = grabColor(QPoint(i - x0, j - y0), false);
118 #else
119             color = grabColor(QPoint(i, j));
120 #endif
121             sumR += color.red();
122             sumG += color.green();
123             sumB += color.blue();
124         }
125     }
126
127 #ifdef Q_WS_X11
128     XDestroyImage(m_image);
129     m_image = 0;
130 #endif
131
132     int numPixel = (x1 - x0) * (y1 - y0);
133     return QColor(sumR / numPixel, sumG / numPixel, sumB / numPixel);
134 }
135
136 void ColorPickerWidget::mousePressEvent(QMouseEvent* event)
137 {
138     if (event->button() != Qt::LeftButton) {
139         closeEventFilter();
140         event->accept();
141         return;
142     }
143     QWidget::mousePressEvent(event);
144 }
145
146 void ColorPickerWidget::mouseReleaseEvent(QMouseEvent *event)
147 {
148     if (m_filterActive) {
149         closeEventFilter();
150
151         if (m_size->value() == 1)
152             emit colorPicked(grabColor(event->globalPos()));
153         else
154             emit colorPicked(averagePickedColor(event->globalPos()));
155
156         return;
157     }
158     QWidget::mouseReleaseEvent(event);
159 }
160
161 void ColorPickerWidget::keyPressEvent(QKeyEvent *event)
162 {
163     if (m_filterActive) {
164         // "special keys" (non letter, numeral) do not work, so close for every key
165         //if (event->key() == Qt::Key_Escape)
166         closeEventFilter();
167         event->accept();
168         return;
169     }
170     QWidget::keyPressEvent(event);
171 }
172
173 void ColorPickerWidget::slotSetupEventFilter()
174 {
175     m_filterActive = true;
176 #ifdef Q_WS_X11
177     m_filter = new KCDPickerFilter(this);
178     kapp->installX11EventFilter(m_filter);
179 #endif
180     if (m_size->value() == 1)
181         grabMouse(QCursor(KIcon("color-picker").pixmap(22, 22), 0, 21));
182     else
183         grabMouse(Qt::CrossCursor);
184     grabKeyboard();
185 }
186
187 void ColorPickerWidget::closeEventFilter()
188 {
189     m_filterActive = false;
190 #ifdef Q_WS_X11
191     kapp->removeX11EventFilter(m_filter);
192     delete m_filter;
193     m_filter = 0;
194 #endif
195     releaseMouse();
196     releaseKeyboard();
197 }
198
199 QColor ColorPickerWidget::grabColor(const QPoint &p, bool destroyImage)
200 {
201 #ifdef Q_WS_X11
202     /*
203      we use the X11 API directly in this case as we are not getting back a valid
204      return from QPixmap::grabWindow in the case where the application is using
205      an argb visual
206     */
207     if( !qApp->desktop()->geometry().contains( p ))
208         return QColor();
209     unsigned long xpixel;
210     if (m_image == 0) {
211         Window root = RootWindow(QX11Info::display(), QX11Info::appScreen());
212         m_image = XGetImage(QX11Info::display(), root, p.x(), p.y(), 1, 1, -1, ZPixmap);
213         xpixel = XGetPixel(m_image, 0, 0);
214     } else {
215         xpixel = XGetPixel(m_image, p.x(), p.y());
216     }
217     if (destroyImage) {
218         XDestroyImage(m_image);
219         m_image = 0;
220     }
221     XColor xcol;
222     xcol.pixel = xpixel;
223     xcol.flags = DoRed | DoGreen | DoBlue;
224     XQueryColor(QX11Info::display(),
225                 DefaultColormap(QX11Info::display(), QX11Info::appScreen()),
226                 &xcol);
227     return QColor::fromRgbF(xcol.red / 65535.0, xcol.green / 65535.0, xcol.blue / 65535.0);
228 #else
229     QWidget *desktop = QApplication::desktop();
230     QPixmap pm = QPixmap::grabWindow(desktop->winId(), p.x(), p.y(), 1, 1);
231     QImage i = pm.toImage();
232     return i.pixel(0, 0);
233 #endif
234 }
235
236 #include "colorpickerwidget.moc"