]> git.sesse.net Git - kdenlive/blob - src/colorpickerwidget.cpp
Color Picker:
[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 #endif
63     m_image = 0;
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 qMin here, as we might run into troubles with the cursor otherwise.
75     m_size->setMaximum(qMin(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     int numPixel = (x1 - x0) * (y1 - y0);
100
101     int sumR = 0;
102     int sumG = 0;
103     int sumB = 0;
104
105     // only show message for size > 200 because for smaller values it slows down to much
106     if (size > 200)
107         emit displayMessage(i18n("Requesting color information..."), 0);
108
109     /*
110      Only getting the image once for the whole rect
111      results in a vast speed improvement.
112     */
113 #ifdef Q_WS_X11
114     Window root = RootWindow(QX11Info::display(), QX11Info::appScreen());
115     m_image = XGetImage(QX11Info::display(), root, x0, y0, x1 - x0, y1 - y0, -1, ZPixmap);
116 #else
117     QWidget *desktop = QApplication::desktop();
118     m_image = QPixmap::grabWindow(desktop->winId(), x0, y0, x1 - x0, y1 - y0).toImage();
119 #endif
120
121     for (int i = x0; i <= x1; ++i) {
122         for (int j = y0; j <= y1; ++j) {
123             QColor color;
124             color = grabColor(QPoint(i - x0, j - y0), false);
125             sumR += color.red();
126             sumG += color.green();
127             sumB += color.blue();
128         }
129
130         // Warning: slows things down, so don't do it for every pixel (the inner for loop)
131         if (size > 200)
132             emit displayMessage(i18n("Requesting color information..."), (int)(((i - x0) * (y1 - y0)) / (qreal)numPixel * 100));
133     }
134
135 #ifdef Q_WS_X11
136     XDestroyImage(m_image);
137 #endif
138     m_image = 0;
139
140     if (size > 200)
141         emit displayMessage(i18n("Calculated average color for rectangle."), -1);
142
143     return QColor(sumR / numPixel, sumG / numPixel, sumB / numPixel);
144 }
145
146 void ColorPickerWidget::mousePressEvent(QMouseEvent* event)
147 {
148     if (event->button() != Qt::LeftButton) {
149         closeEventFilter();
150         event->accept();
151         return;
152     }
153     QWidget::mousePressEvent(event);
154 }
155
156 void ColorPickerWidget::mouseReleaseEvent(QMouseEvent *event)
157 {
158     if (m_filterActive) {
159         closeEventFilter();
160
161         if (m_size->value() == 1)
162             emit colorPicked(grabColor(event->globalPos()));
163         else
164             emit colorPicked(averagePickedColor(event->globalPos()));
165
166         return;
167     }
168     QWidget::mouseReleaseEvent(event);
169 }
170
171 void ColorPickerWidget::keyPressEvent(QKeyEvent *event)
172 {
173     if (m_filterActive) {
174         // "special keys" (non letter, numeral) do not work, so close for every key
175         //if (event->key() == Qt::Key_Escape)
176         closeEventFilter();
177         event->accept();
178         return;
179     }
180     QWidget::keyPressEvent(event);
181 }
182
183 void ColorPickerWidget::slotSetupEventFilter()
184 {
185     m_filterActive = true;
186 #ifdef Q_WS_X11
187     m_filter = new KCDPickerFilter(this);
188     kapp->installX11EventFilter(m_filter);
189 #endif
190     if (m_size->value() < 10)
191         grabMouse(QCursor(KIcon("color-picker").pixmap(22, 22), 0, 21));
192     else
193         grabMouse(QCursor(KIcon("kdenlive-select-all").pixmap(m_size->value(), m_size->value())));
194     grabKeyboard();
195 }
196
197 void ColorPickerWidget::closeEventFilter()
198 {
199     m_filterActive = false;
200 #ifdef Q_WS_X11
201     kapp->removeX11EventFilter(m_filter);
202     delete m_filter;
203     m_filter = 0;
204 #endif
205     releaseMouse();
206     releaseKeyboard();
207 }
208
209 QColor ColorPickerWidget::grabColor(const QPoint &p, bool destroyImage)
210 {
211 #ifdef Q_WS_X11
212     /*
213      we use the X11 API directly in this case as we are not getting back a valid
214      return from QPixmap::grabWindow in the case where the application is using
215      an argb visual
216     */
217     if( !qApp->desktop()->geometry().contains( p ))
218         return QColor();
219     unsigned long xpixel;
220     if (m_image == 0) {
221         Window root = RootWindow(QX11Info::display(), QX11Info::appScreen());
222         m_image = XGetImage(QX11Info::display(), root, p.x(), p.y(), 1, 1, -1, ZPixmap);
223         xpixel = XGetPixel(m_image, 0, 0);
224     } else {
225         xpixel = XGetPixel(m_image, p.x(), p.y());
226     }
227     if (destroyImage) {
228         XDestroyImage(m_image);
229         m_image = 0;
230     }
231     XColor xcol;
232     xcol.pixel = xpixel;
233     xcol.flags = DoRed | DoGreen | DoBlue;
234     XQueryColor(QX11Info::display(),
235                 DefaultColormap(QX11Info::display(), QX11Info::appScreen()),
236                 &xcol);
237     return QColor::fromRgbF(xcol.red / 65535.0, xcol.green / 65535.0, xcol.blue / 65535.0);
238 #else
239     if (m_image == 0) {
240         QWidget *desktop = QApplication::desktop();
241         QPixmap pm = QPixmap::grabWindow(desktop->winId(), p.x(), p.y(), 1, 1);
242         QImage i = pm.toImage();
243         return i.pixel(0, 0);
244     } else {
245         return m_image.pixel(p.x(), p.y());
246     }
247 #endif
248 }
249
250 #include "colorpickerwidget.moc"