1 /***************************************************************************
2 * Copyright (C) 2012 by Peter Penz <peter.penz19@gmail.com> *
4 * Based on KFilePlacesItem from kdelibs: *
5 * Copyright (C) 2007 Kevin Ottens <ervin@kde.org> *
7 * This program is free software; you can redistribute it and/or modify *
8 * it under the terms of the GNU General Public License as published by *
9 * the Free Software Foundation; either version 2 of the License, or *
10 * (at your option) any later version. *
12 * This program is distributed in the hope that it will be useful, *
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
15 * GNU General Public License for more details. *
17 * You should have received a copy of the GNU General Public License *
18 * along with this program; if not, write to the *
19 * Free Software Foundation, Inc., *
20 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA *
21 ***************************************************************************/
23 #include "placesitem.h"
25 #include <KBookmarkManager>
26 #include "dolphindebug.h"
28 #include <KLocalizedString>
29 #include "placesitemsignalhandler.h"
31 #include <Solid/Block>
33 PlacesItem::PlacesItem(const KBookmark
& bookmark
, PlacesItem
* parent
) :
34 KStandardItem(parent
),
40 m_signalHandler(nullptr),
41 m_trashDirLister(nullptr),
44 m_signalHandler
= new PlacesItemSignalHandler(this);
45 setBookmark(bookmark
);
48 PlacesItem::~PlacesItem()
50 delete m_signalHandler
;
51 delete m_trashDirLister
;
54 void PlacesItem::setUrl(const QUrl
&url
)
56 // The default check in KStandardItem::setDataValue()
57 // for equal values does not work with a custom value
58 // like QUrl. Hence do a manual check to prevent that
59 // setting an equal URL results in an itemsChanged()
61 if (dataValue("url").toUrl() != url
) {
62 delete m_trashDirLister
;
63 if (url
.scheme() == QLatin1String("trash")) {
64 // The trash icon must always be updated dependent on whether
65 // the trash is empty or not. We use a KDirLister that automatically
66 // watches for changes if the number of items has been changed.
67 // The update of the icon is handled in onTrashDirListerCompleted().
68 m_trashDirLister
= new KDirLister();
69 m_trashDirLister
->setAutoErrorHandlingEnabled(false, nullptr);
70 m_trashDirLister
->setDelayedMimeTypes(true);
71 QObject::connect(m_trashDirLister
.data(), static_cast<void(KDirLister::*)()>(&KDirLister::completed
),
72 m_signalHandler
.data(), &PlacesItemSignalHandler::onTrashDirListerCompleted
);
73 m_trashDirLister
->openUrl(url
);
76 setDataValue("url", url
);
80 QUrl
PlacesItem::url() const
82 return dataValue("url").toUrl();
85 void PlacesItem::setUdi(const QString
& udi
)
87 setDataValue("udi", udi
);
90 QString
PlacesItem::udi() const
92 return dataValue("udi").toString();
95 void PlacesItem::setHidden(bool hidden
)
97 setDataValue("isHidden", hidden
);
100 bool PlacesItem::isHidden() const
102 return dataValue("isHidden").toBool();
105 void PlacesItem::setSystemItem(bool isSystemItem
)
107 setDataValue("isSystemItem", isSystemItem
);
110 bool PlacesItem::isSystemItem() const
112 return dataValue("isSystemItem").toBool();
115 Solid::Device
PlacesItem::device() const
120 void PlacesItem::setBookmark(const KBookmark
& bookmark
)
122 if (bookmark
== m_bookmark
) {
126 m_bookmark
= bookmark
;
133 const QString udi
= bookmark
.metaDataItem(QStringLiteral("UDI"));
135 setIcon(bookmark
.icon());
136 setText(i18nc("KFile System Bookmarks", bookmark
.text().toUtf8().constData()));
137 setUrl(bookmark
.url());
138 setSystemItem(bookmark
.metaDataItem(QStringLiteral("isSystemItem")) == QLatin1String("true"));
140 initializeDevice(udi
);
143 setHidden(bookmark
.metaDataItem(QStringLiteral("IsHidden")) == QLatin1String("true"));
146 KBookmark
PlacesItem::bookmark() const
151 bool PlacesItem::storageSetupNeeded() const
153 return m_access
? !m_access
->isAccessible() : false;
156 bool PlacesItem::isSearchOrTimelineUrl() const
158 const QString urlScheme
= url().scheme();
159 return (urlScheme
.contains("search") || urlScheme
.contains("timeline"));
162 void PlacesItem::onDataValueChanged(const QByteArray
& role
,
163 const QVariant
& current
,
164 const QVariant
& previous
)
169 if (!m_bookmark
.isNull()) {
170 updateBookmarkForRole(role
);
174 void PlacesItem::onDataChanged(const QHash
<QByteArray
, QVariant
>& current
,
175 const QHash
<QByteArray
, QVariant
>& previous
)
179 if (!m_bookmark
.isNull()) {
180 QHashIterator
<QByteArray
, QVariant
> it(current
);
181 while (it
.hasNext()) {
183 updateBookmarkForRole(it
.key());
188 void PlacesItem::initializeDevice(const QString
& udi
)
190 m_device
= Solid::Device(udi
);
191 if (!m_device
.isValid()) {
195 m_access
= m_device
.as
<Solid::StorageAccess
>();
196 m_volume
= m_device
.as
<Solid::StorageVolume
>();
197 m_disc
= m_device
.as
<Solid::OpticalDisc
>();
198 m_mtp
= m_device
.as
<Solid::PortableMediaPlayer
>();
200 setText(m_device
.description());
201 setIcon(m_device
.icon());
202 setIconOverlays(m_device
.emblems());
206 setUrl(QUrl::fromLocalFile(m_access
->filePath()));
207 QObject::connect(m_access
.data(), &Solid::StorageAccess::accessibilityChanged
,
208 m_signalHandler
.data(), &PlacesItemSignalHandler::onAccessibilityChanged
);
209 QObject::connect(m_access
.data(), &Solid::StorageAccess::teardownRequested
,
210 m_signalHandler
.data(), &PlacesItemSignalHandler::onTearDownRequested
);
211 } else if (m_disc
&& (m_disc
->availableContent() & Solid::OpticalDisc::Audio
) != 0) {
212 Solid::Block
*block
= m_device
.as
<Solid::Block
>();
214 const QString device
= block
->device();
215 setUrl(QUrl(QStringLiteral("audiocd:/?device=%1").arg(device
)));
217 setUrl(QUrl(QStringLiteral("audiocd:/")));
220 setUrl(QUrl(QStringLiteral("mtp:udi=%1").arg(m_device
.udi())));
224 void PlacesItem::onAccessibilityChanged()
226 setIconOverlays(m_device
.emblems());
227 setUrl(QUrl::fromLocalFile(m_access
->filePath()));
230 void PlacesItem::onTrashDirListerCompleted()
232 Q_ASSERT(url().scheme() == QLatin1String("trash"));
234 const bool isTrashEmpty
= m_trashDirLister
->items().isEmpty();
235 setIcon(isTrashEmpty
? QStringLiteral("user-trash") : QStringLiteral("user-trash-full"));
238 void PlacesItem::updateBookmarkForRole(const QByteArray
& role
)
240 Q_ASSERT(!m_bookmark
.isNull());
241 if (role
== "iconName") {
242 m_bookmark
.setIcon(icon());
243 } else if (role
== "text") {
244 // Only store the text in the KBookmark if it is not the translation of
245 // the current text. This makes sure that the text is re-translated if
246 // the user chooses another language, or the translation itself changes.
248 // NOTE: It is important to use "KFile System Bookmarks" as context
249 // (see PlacesItemModel::createSystemBookmarks()).
250 if (text() != i18nc("KFile System Bookmarks", m_bookmark
.text().toUtf8().data())) {
251 m_bookmark
.setFullText(text());
253 } else if (role
== "url") {
254 m_bookmark
.setUrl(url());
255 } else if (role
== "udi") {
256 m_bookmark
.setMetaDataItem(QStringLiteral("UDI"), udi());
257 } else if (role
== "isSystemItem") {
258 m_bookmark
.setMetaDataItem(QStringLiteral("isSystemItem"), isSystemItem() ? QStringLiteral("true") : QStringLiteral("false"));
259 } else if (role
== "isHidden") {
260 m_bookmark
.setMetaDataItem(QStringLiteral("IsHidden"), isHidden() ? QStringLiteral("true") : QStringLiteral("false"));
264 QString
PlacesItem::generateNewId()
266 // The ID-generation must be different as done in KFilePlacesItem from kdelibs
267 // to prevent identical IDs, because 'count' is of course not shared. We append a
268 // " (V2)" to indicate that the ID has been generated by
269 // a new version of the places view.
270 static int count
= 0;
271 return QString::number(QDateTime::currentDateTimeUtc().toTime_t()) +
272 '/' + QString::number(count
++) + " (V2)";
275 PlacesItemSignalHandler
*PlacesItem::signalHandler() const
277 return m_signalHandler
.data();