1 /***************************************************************************
2 * Copyright (C) 2006 by Peter Penz <peter.penz@gmx.at> *
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. *
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. *
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 ***************************************************************************/
20 #include "infosidebarpage.h"
27 #include <qpushbutton.h>
29 #include <q3vgroupbox.h>
30 #include <q3popupmenu.h>
32 #include <qfontmetrics.h>
34 #include <q3hgroupbox.h>
36 #include <Q3ValueList>
38 #include <Q3VBoxLayout>
40 #include <kbookmarkmanager.h>
42 #include <kstandarddirs.h>
43 #include <kio/previewjob.h>
44 #include <kfileitem.h>
46 #include <kglobalsettings.h>
47 #include <kfilemetainfo.h>
50 #include "dolphinmainwindow.h"
51 #include "pixmapviewer.h"
52 #include "dolphinsettings.h"
54 InfoSidebarPage::InfoSidebarPage(DolphinMainWindow
* mainWindow
, QWidget
* parent
) :
55 SidebarPage(mainWindow
, parent
),
56 m_multipleSelection(false),
57 m_pendingPreview(false),
65 const int spacing
= KDialog::spacingHint();
67 m_timer
= new QTimer(this);
68 connect(m_timer
, SIGNAL(timeout()),
69 this, SLOT(slotTimeout()));
71 Q3VBoxLayout
* layout
= new Q3VBoxLayout(this);
72 layout
->setSpacing(spacing
);
75 m_preview
= new PixmapViewer(this);
76 m_preview
->setMinimumWidth(K3Icon::SizeEnormous
);
77 m_preview
->setFixedHeight(K3Icon::SizeEnormous
);
80 m_name
= new QLabel(this);
81 m_name
->setTextFormat(Qt::RichText
);
82 m_name
->setAlignment(m_name
->alignment() | Qt::AlignHCenter
);
83 QFontMetrics
fontMetrics(m_name
->font());
84 m_name
->setMinimumHeight(fontMetrics
.height() * 3);
85 m_name
->setSizePolicy(QSizePolicy::MinimumExpanding
, QSizePolicy::Maximum
);
87 QWidget
* sep1
= new Q3HGroupBox(this); // TODO: check whether default widget exist for this?
88 sep1
->setFixedHeight(1);
90 // general information
91 m_infoGrid
= new Q3Grid(2, this);
92 m_infoGrid
->setSizePolicy(QSizePolicy::Preferred
, QSizePolicy::Fixed
);
94 QWidget
* sep2
= new Q3HGroupBox(this); // TODO: check whether default widget exist for this?
95 sep2
->setFixedHeight(1);
98 m_actionBox
= new KVBox(this);
99 m_actionBox
->setSizePolicy(QSizePolicy::Preferred
, QSizePolicy::Fixed
);
101 // Add a dummy widget with no restriction regarding a vertical resizing.
102 // This assures that information is always top aligned.
103 QWidget
* dummy
= new QWidget(this);
105 layout
->addItem(new QSpacerItem(spacing
, spacing
, QSizePolicy::Preferred
, QSizePolicy::Fixed
));
106 layout
->addWidget(m_preview
);
107 layout
->addWidget(m_name
);
108 layout
->addWidget(sep1
);
109 layout
->addWidget(m_infoGrid
);
110 layout
->addWidget(sep2
);
111 layout
->addWidget(m_actionBox
);
112 layout
->addWidget(dummy
);
114 connect(mainWindow
, SIGNAL(selectionChanged()),
115 this, SLOT(showItemInfo()));
117 connectToActiveView();
120 InfoSidebarPage::~InfoSidebarPage()
124 void InfoSidebarPage::activeViewChanged()
126 connectToActiveView();
129 void InfoSidebarPage::requestDelayedItemInfo(const KUrl
& url
)
133 if (!url
.isEmpty() && !m_multipleSelection
) {
134 m_urlCandidate
= url
;
135 m_timer
->setSingleShot(true);
140 void InfoSidebarPage::requestItemInfo(const KUrl
& url
)
144 if (!url
.isEmpty() && !m_multipleSelection
) {
150 void InfoSidebarPage::showItemInfo()
154 m_multipleSelection
= false;
156 // show the preview...
157 DolphinView
* view
= mainWindow()->activeView();
158 const KFileItemList selectedItems
= view
->selectedItems();
159 if (selectedItems
.count() > 1) {
160 m_multipleSelection
= true;
163 if (m_multipleSelection
) {
164 KIconLoader iconLoader
;
165 QPixmap icon
= iconLoader
.loadIcon("exec",
167 K3Icon::SizeEnormous
);
168 m_preview
->setPixmap(icon
);
169 m_name
->setText(i18n("%1 items selected",selectedItems
.count()));
171 else if (!applyBookmark()) {
172 // try to get a preview pixmap from the item...
174 list
.append(m_shownUrl
);
176 m_pendingPreview
= true;
177 m_preview
->setPixmap(QPixmap());
179 KIO::PreviewJob
* job
= KIO::filePreview(list
,
181 K3Icon::SizeEnormous
);
182 connect(job
, SIGNAL(gotPreview(const KFileItem
*, const QPixmap
&)),
183 this, SLOT(gotPreview(const KFileItem
*, const QPixmap
&)));
184 connect(job
, SIGNAL(failed(const KFileItem
*)),
185 this, SLOT(slotPreviewFailed(const KFileItem
*)));
188 text
.append(m_shownUrl
.fileName());
190 m_name
->setText(text
);
197 void InfoSidebarPage::slotTimeout()
199 m_shownUrl
= m_urlCandidate
;
203 void InfoSidebarPage::slotPreviewFailed(const KFileItem
* item
)
205 m_pendingPreview
= false;
206 if (!applyBookmark()) {
207 m_preview
->setPixmap(item
->pixmap(K3Icon::SizeEnormous
));
211 void InfoSidebarPage::gotPreview(const KFileItem
* /* item */,
212 const QPixmap
& pixmap
)
214 if (m_pendingPreview
) {
215 m_preview
->setPixmap(pixmap
);
216 m_pendingPreview
= false;
220 void InfoSidebarPage::startService(int index
)
222 DolphinView
* view
= mainWindow()->activeView();
223 if (view
->hasSelection()) {
224 KUrl::List selectedUrls
= view
->selectedUrls();
225 KDEDesktopMimeType::executeService(selectedUrls
, m_actionsVector
[index
]);
228 KDEDesktopMimeType::executeService(m_shownUrl
, m_actionsVector
[index
]);
232 void InfoSidebarPage::connectToActiveView()
236 DolphinView
* view
= mainWindow()->activeView();
237 connect(view
, SIGNAL(requestItemInfo(const KUrl
&)),
238 this, SLOT(requestDelayedItemInfo(const KUrl
&)));
239 connect(view
, SIGNAL(urlChanged(const KUrl
&)),
240 this, SLOT(requestItemInfo(const KUrl
&)));
242 m_shownUrl
= view
->url();
246 bool InfoSidebarPage::applyBookmark()
248 KBookmarkGroup root
= DolphinSettings::instance().bookmarkManager()->root();
249 KBookmark bookmark
= root
.first();
250 while (!bookmark
.isNull()) {
251 if (m_shownUrl
.equals(bookmark
.url(), KUrl::CompareWithoutTrailingSlash
)) {
253 text
.append(bookmark
.text());
255 m_name
->setText(text
);
257 KIconLoader iconLoader
;
258 QPixmap icon
= iconLoader
.loadIcon(bookmark
.icon(),
260 K3Icon::SizeEnormous
);
261 m_preview
->setPixmap(icon
);
264 bookmark
= root
.next(bookmark
);
270 void InfoSidebarPage::cancelRequest()
273 m_pendingPreview
= false;
276 void InfoSidebarPage::createMetaInfo()
278 // To prevent a flickering it's important to reuse available
279 // labels instead of deleting them and recreate them afterwards.
280 // The methods beginInfoLines(), addInfoLine() and endInfoLines()
281 // take care of this.
283 DolphinView
* view
= mainWindow()->activeView();
284 if (!view
->hasSelection()) {
285 KFileItem
fileItem(S_IFDIR
, KFileItem::Unknown
, m_shownUrl
);
288 if (fileItem
.isDir()) {
289 addInfoLine(i18n("Type:"), i18n("Directory"));
292 addInfoLine(i18n("Type:"), fileItem
.mimeComment());
294 QString
sizeText(KIO::convertSize(fileItem
.size()));
295 addInfoLine(i18n("Size:"), sizeText
);
296 addInfoLine(i18n("Modified:"), fileItem
.timeString());
298 const KFileMetaInfo
& metaInfo
= fileItem
.metaInfo();
299 if (metaInfo
.isValid()) {
300 QStringList keys
= metaInfo
.supportedKeys();
301 for (QStringList::Iterator it
= keys
.begin(); it
!= keys
.end(); ++it
) {
302 if (showMetaInfo(*it
)) {
303 KFileMetaInfoItem metaInfoItem
= metaInfo
.item(*it
);
304 addInfoLine(*it
, metaInfoItem
.string());
313 void InfoSidebarPage::beginInfoLines()
315 m_currInfoLineIdx
= 0;
318 void InfoSidebarPage::endInfoLines()
320 if (m_currInfoLineIdx
<= 0) {
324 // remove labels which have not been used
325 if (m_currInfoLineIdx
< static_cast<int>(m_infoWidgets
.count())) {
326 Q3PtrListIterator
<QLabel
> deleteIter(m_infoWidgets
);
327 deleteIter
+= m_currInfoLineIdx
;
331 while ((widget
= deleteIter
.current()) != 0) {
333 widget
->deleteLater();
337 for (int i
= 0; i
< removeCount
; ++i
) {
338 m_infoWidgets
.removeLast();
343 bool InfoSidebarPage::showMetaInfo(const QString
& key
) const
345 // sorted list of keys, where it's data should be shown
346 static const char* keys
[] = {
361 // do a binary search for the key...
363 int bottom
= sizeof(keys
) / sizeof(char*) - 1;
364 while (top
< bottom
) {
365 const int middle
= (top
+ bottom
) / 2;
366 const int result
= key
.compare(keys
[middle
]);
370 else if (result
> 0) {
381 void InfoSidebarPage::addInfoLine(const QString
& labelText
, const QString
& infoText
)
383 QString
labelStr("<b>");
384 labelStr
.append(labelText
);
385 labelStr
.append("</b> ");
387 const int count
= m_infoWidgets
.count();
388 if (m_currInfoLineIdx
< count
- 1) {
389 // reuse available labels
390 m_infoWidgets
.at(m_currInfoLineIdx
++)->setText(labelStr
);
391 m_infoWidgets
.at(m_currInfoLineIdx
++)->setText(infoText
);
394 // no labels are available anymore, hence create 2 new ones
395 QLabel
* label
= new QLabel(labelStr
, m_infoGrid
);
396 label
->setTextFormat(Qt::RichText
);
397 label
->setAlignment(Qt::AlignRight
|
400 m_infoWidgets
.append(label
);
402 QLabel
* info
= new QLabel(infoText
, m_infoGrid
);
403 info
->setTextFormat(Qt::RichText
);
404 info
->setAlignment(Qt::AlignTop
| Qt::TextWordWrap
);
406 m_infoWidgets
.append(info
);
408 m_currInfoLineIdx
+= 2;
412 void InfoSidebarPage::insertActions()
414 // delete all existing action widgets
415 // TODO: just use children() from QObject...
416 Q3PtrListIterator
<QWidget
> deleteIter(m_actionWidgets
);
418 while ((widget
= deleteIter
.current()) != 0) {
420 widget
->deleteLater();
424 m_actionWidgets
.clear();
425 m_actionsVector
.clear();
427 int actionsIndex
= 0;
429 // The algorithm for searching the available actions works on a list
430 // of KFileItems. If no selection is given, a temporary KFileItem
431 // by the given Url 'url' is created and added to the list.
432 KFileItem
fileItem(S_IFDIR
, KFileItem::Unknown
, m_shownUrl
);
433 KFileItemList itemList
= mainWindow()->activeView()->selectedItems();
434 if (itemList
.isEmpty()) {
436 itemList
.append(&fileItem
);
439 // 'itemList' contains now all KFileItems, where an item information should be shown.
440 // TODO: the following algorithm is quite equal to DolphinContextMenu::insertActionItems().
441 // It's open yet whether they should be merged or whether they have to work slightly different.
442 QStringList dirs
= KGlobal::dirs()->findDirs("data", "dolphin/servicemenus/");
443 for (QStringList::ConstIterator dirIt
= dirs
.begin(); dirIt
!= dirs
.end(); ++dirIt
) {
445 QStringList entries
= dir
.entryList("*.desktop", QDir::Files
);
447 for (QStringList::ConstIterator entryIt
= entries
.begin(); entryIt
!= entries
.end(); ++entryIt
) {
448 KSimpleConfig
cfg(*dirIt
+ *entryIt
, true);
449 cfg
.setDesktopGroup();
450 if ((cfg
.hasKey("Actions") || cfg
.hasKey("X-KDE-GetActionMenu")) && cfg
.hasKey("ServiceTypes")) {
451 const QStringList types
= cfg
.readListEntry("ServiceTypes", ',');
452 for (QStringList::ConstIterator it
= types
.begin(); it
!= types
.end(); ++it
) {
453 // check whether the mime type is equal or whether the
454 // mimegroup (e. g. image/*) is supported
457 if ((*it
) == "all/allfiles") {
458 // The service type is valid for all files, but not for directories.
459 // Check whether the selected items only consist of files...
460 QListIterator
<KFileItem
*> mimeIt(itemList
);
462 while (insert
&& mimeIt
.hasNext()) {
463 KFileItem
* item
= mimeIt
.next();
464 insert
= !item
->isDir();
469 // Check whether the MIME types of all selected files match
470 // to the mimetype of the service action. As soon as one MIME
471 // type does not match, no service menu is shown at all.
472 QListIterator
<KFileItem
*> mimeIt(itemList
);
474 while (insert
&& mimeIt
.hasNext()) {
475 KFileItem
* item
= mimeIt
.next();
476 const QString
mimeType(item
->mimetype());
477 const QString
mimeGroup(mimeType
.left(mimeType
.indexOf('/')));
479 insert
= (*it
== mimeType
) ||
480 ((*it
).right(1) == "*") &&
481 ((*it
).left((*it
).indexOf('/')) == mimeGroup
);
486 const QString submenuName
= cfg
.readEntry( "X-KDE-Submenu" );
487 Q3PopupMenu
* popup
= 0;
488 if (!submenuName
.isEmpty()) {
489 // create a sub menu containing all actions
490 popup
= new Q3PopupMenu();
491 connect(popup
, SIGNAL(activated(int)),
492 this, SLOT(startService(int)));
494 QPushButton
* button
= new QPushButton(submenuName
, m_actionBox
);
495 button
->setFlat(true);
496 button
->setMenu(popup
);
498 m_actionWidgets
.append(button
);
501 Q3ValueList
<KDEDesktopMimeType::Service
> userServices
=
502 KDEDesktopMimeType::userDefinedServices(*dirIt
+ *entryIt
, true);
504 // iterate through all actions and add them to a widget
505 Q3ValueList
<KDEDesktopMimeType::Service
>::Iterator serviceIt
;
506 for (serviceIt
= userServices
.begin(); serviceIt
!= userServices
.end(); ++serviceIt
) {
507 KDEDesktopMimeType::Service service
= (*serviceIt
);
509 ServiceButton
* button
= new ServiceButton(SmallIcon(service
.m_strIcon
),
513 connect(button
, SIGNAL(requestServiceStart(int)),
514 this, SLOT(startService(int)));
515 m_actionWidgets
.append(button
);
519 popup
->insertItem(SmallIcon(service
.m_strIcon
), service
.m_strName
, actionsIndex
);
522 m_actionsVector
.append(service
);
532 ServiceButton::ServiceButton(const QIcon
& icon
,
536 QPushButton(icon
, text
, parent
),
540 setEraseColor(palette().brush(QPalette::Background
).color());
541 setFocusPolicy(Qt::NoFocus
);
542 connect(this, SIGNAL(released()),
543 this, SLOT(slotReleased()));
546 ServiceButton::~ServiceButton()
550 void ServiceButton::paintEvent(QPaintEvent
* event
)
552 QPainter
painter(this);
553 const int buttonWidth
= width();
554 const int buttonHeight
= height();
556 QColor backgroundColor
;
557 QColor foregroundColor
;
559 backgroundColor
= KGlobalSettings::highlightColor();
560 foregroundColor
= KGlobalSettings::highlightedTextColor();
563 backgroundColor
= palette().brush(QPalette::Background
).color();
564 foregroundColor
= KGlobalSettings::buttonTextColor();
567 // draw button background
568 painter
.setPen(Qt::NoPen
);
569 painter
.setBrush(backgroundColor
);
570 painter
.drawRect(0, 0, buttonWidth
, buttonHeight
);
572 const int spacing
= KDialog::spacingHint();
576 const int y
= (buttonHeight
- K3Icon::SizeSmall
) / 2;
577 const QIcon
* set
= iconSet();
579 painter
.drawPixmap(x
, y
, set
->pixmap(QIcon::Small
, QIcon::Normal
));
581 x
+= K3Icon::SizeSmall
+ spacing
;
584 painter
.setPen(foregroundColor
);
586 const int textWidth
= buttonWidth
- x
;
587 QFontMetrics
fontMetrics(font());
588 const bool clipped
= fontMetrics
.width(text()) >= textWidth
;
589 //const int align = clipped ? Qt::AlignVCenter : Qt::AlignCenter;
590 painter
.drawText(QRect(x
, 0, textWidth
, buttonHeight
), Qt::AlignVCenter
, text());
593 // Blend the right area of the text with the background, as the
595 // TODO #1: use alpha blending in Qt4 instead of drawing the text that often
596 // TODO #2: same code as in UrlNavigatorButton::drawButton() -> provide helper class?
597 const int blendSteps
= 16;
599 QColor
blendColor(backgroundColor
);
600 const int redInc
= (foregroundColor
.red() - backgroundColor
.red()) / blendSteps
;
601 const int greenInc
= (foregroundColor
.green() - backgroundColor
.green()) / blendSteps
;
602 const int blueInc
= (foregroundColor
.blue() - backgroundColor
.blue()) / blendSteps
;
603 for (int i
= 0; i
< blendSteps
; ++i
) {
604 painter
.setClipRect(QRect(x
+ textWidth
- i
, 0, 1, buttonHeight
));
605 painter
.setPen(blendColor
);
606 painter
.drawText(QRect(x
, 0, textWidth
, buttonHeight
), Qt::AlignVCenter
, text());
608 blendColor
.setRgb(blendColor
.red() + redInc
,
609 blendColor
.green() + greenInc
,
610 blendColor
.blue() + blueInc
);
615 void ServiceButton::enterEvent(QEvent
* event
)
617 QPushButton::enterEvent(event
);
622 void ServiceButton::leaveEvent(QEvent
* event
)
624 QPushButton::leaveEvent(event
);
629 void ServiceButton::slotReleased()
631 emit
requestServiceStart(m_index
);
634 #include "infosidebarpage.moc"