2 This file is part of the KDE project
3 SPDX-FileCopyrightText: 2020 Felix Ernst <fe.a.ernst@gmail.com>
5 SPDX-License-Identifier: LGPL-2.1-only OR LGPL-3.0-only OR LicenseRef-KDE-Accepted-LGPL
8 #include "dolphinnavigatorswidgetaction.h"
10 #include "trash/dolphintrash.h"
12 #include <KLocalizedString>
13 #include <KNotificationJobUiDelegate>
15 #include <KXMLGUIFactory>
16 #include <KXmlGuiWindow>
18 #include <KIO/ApplicationLauncherJob>
20 #include <QApplication>
21 #include <QDomDocument>
22 #include <QHBoxLayout>
23 #include <QPushButton>
29 DolphinNavigatorsWidgetAction::DolphinNavigatorsWidgetAction(QWidget
*parent
) :
30 QWidgetAction
{parent
},
31 m_splitter
{new QSplitter(Qt::Horizontal
)},
32 m_adjustSpacingTimer
{new QTimer(this)},
33 m_globalXOfSplitter
{INT_MIN
},
34 m_globalXOfPrimary
{INT_MIN
},
35 m_widthOfPrimary
{INT_MIN
},
36 m_globalXOfSecondary
{INT_MIN
},
37 m_widthOfSecondary
{INT_MIN
}
40 setIcon(QIcon::fromTheme(QStringLiteral("dialog-scripts")));
42 m_splitter
->setChildrenCollapsible(false);
44 m_splitter
->addWidget(createNavigatorWidget(Primary
));
46 m_adjustSpacingTimer
->setInterval(100);
47 m_adjustSpacingTimer
->setSingleShot(true);
48 connect(m_adjustSpacingTimer
.get(), &QTimer::timeout
,
49 this, &DolphinNavigatorsWidgetAction::adjustSpacing
);
52 void DolphinNavigatorsWidgetAction::createSecondaryUrlNavigator()
54 Q_ASSERT(m_splitter
->count() == 1);
55 m_splitter
->addWidget(createNavigatorWidget(Secondary
));
56 Q_ASSERT(m_splitter
->count() == 2);
60 void DolphinNavigatorsWidgetAction::followViewContainerGeometry(
61 int globalXOfPrimary
, int widthOfPrimary
)
63 followViewContainersGeometry(globalXOfPrimary
, widthOfPrimary
, INT_MIN
, INT_MIN
);
66 void DolphinNavigatorsWidgetAction::followViewContainersGeometry(
67 int globalXOfPrimary
, int widthOfPrimary
,
68 int globalXOfSecondary
, int widthOfSecondary
)
70 if (QApplication::layoutDirection() == Qt::LeftToRight
) {
71 m_globalXOfSplitter
= m_splitter
->mapToGlobal(QPoint(0,0)).x();
72 m_globalXOfPrimary
= globalXOfPrimary
;
73 m_globalXOfSecondary
= globalXOfSecondary
;
75 // When the direction is reversed, globalX does not change.
76 // For the adjustSpacing() code to work we need globalX to measure from right to left
77 // and to measure up to the rightmost point of a widget instead of the leftmost.
78 m_globalXOfSplitter
= (-1) * (m_splitter
->mapToGlobal(QPoint(0,0)).x() + m_splitter
->width());
79 m_globalXOfPrimary
= (-1) * (globalXOfPrimary
+ widthOfPrimary
);
80 m_globalXOfSecondary
= (globalXOfSecondary
== INT_MIN
) ? INT_MIN
:
81 (-1) * (globalXOfSecondary
+ widthOfSecondary
);
83 m_widthOfPrimary
= widthOfPrimary
;
84 m_widthOfSecondary
= widthOfSecondary
;
88 bool DolphinNavigatorsWidgetAction::isInToolbar() const
90 return qobject_cast
<QToolBar
*>(m_splitter
->parentWidget());
93 DolphinUrlNavigator
* DolphinNavigatorsWidgetAction::primaryUrlNavigator() const
96 return m_splitter
->widget(0)->findChild
<DolphinUrlNavigator
*>();
99 DolphinUrlNavigator
* DolphinNavigatorsWidgetAction::secondaryUrlNavigator() const
101 Q_ASSERT(m_splitter
);
102 if (m_splitter
->count() < 2) {
105 return m_splitter
->widget(1)->findChild
<DolphinUrlNavigator
*>();
108 void DolphinNavigatorsWidgetAction::setSecondaryNavigatorVisible(bool visible
)
111 Q_ASSERT(m_splitter
->count() == 2);
112 m_splitter
->widget(1)->setVisible(true);
113 } else if (m_splitter
->count() > 1) {
114 m_splitter
->widget(1)->setVisible(false);
115 // Fix an unlikely event of wrong trash button visibility.
116 emptyTrashButton(Secondary
)->setVisible(false);
121 QWidget
*DolphinNavigatorsWidgetAction::createWidget(QWidget
*parent
)
123 QWidget
*oldParent
= m_splitter
->parentWidget();
124 if (oldParent
&& oldParent
->layout()) {
125 oldParent
->layout()->removeWidget(m_splitter
.get());
126 QGridLayout
*layout
= qobject_cast
<QGridLayout
*>(oldParent
->layout());
127 if (qobject_cast
<QToolBar
*>(parent
) && layout
) {
128 // in DolphinTabPage::insertNavigatorsWidget the minimumHeight of this row was
129 // set to fit the m_splitter. Since we are now removing it again, the
130 // minimumHeight can be reset to 0.
131 layout
->setRowMinimumHeight(0, 0);
134 m_splitter
->setParent(parent
);
135 return m_splitter
.get();
138 void DolphinNavigatorsWidgetAction::deleteWidget(QWidget
*widget
)
141 m_splitter
->setParent(nullptr);
144 void DolphinNavigatorsWidgetAction::adjustSpacing()
146 Q_ASSERT(m_globalXOfSplitter
!= INT_MIN
);
147 Q_ASSERT(m_globalXOfPrimary
!= INT_MIN
);
148 Q_ASSERT(m_widthOfPrimary
!= INT_MIN
);
149 const int widthOfSplitterPrimary
= m_globalXOfPrimary
+ m_widthOfPrimary
- m_globalXOfSplitter
;
150 const QList
<int> splitterSizes
= {widthOfSplitterPrimary
,
151 m_splitter
->width() - widthOfSplitterPrimary
};
152 m_splitter
->setSizes(splitterSizes
);
154 // primary side of m_splitter
155 int leadingSpacing
= m_globalXOfPrimary
- m_globalXOfSplitter
;
156 if (leadingSpacing
< 0) {
159 int trailingSpacing
= (m_globalXOfSplitter
+ m_splitter
->width())
160 - (m_globalXOfPrimary
+ m_widthOfPrimary
);
161 #if KIO_VERSION < QT_VERSION_CHECK(5, 78, 0)
162 if (trailingSpacing
< 0 || emptyTrashButton(Primary
)->isVisible()) {
164 if (trailingSpacing
< 0 || emptyTrashButton(Primary
)->isVisible()
165 || networkFolderButton(Primary
)->isVisible()
170 const int widthLeftForUrlNavigator
= m_splitter
->widget(0)->width() - leadingSpacing
- trailingSpacing
;
171 const int widthNeededForUrlNavigator
= primaryUrlNavigator()->sizeHint().width() - widthLeftForUrlNavigator
;
172 if (widthNeededForUrlNavigator
> 0) {
173 trailingSpacing
-= widthNeededForUrlNavigator
;
174 if (trailingSpacing
< 0) {
175 leadingSpacing
+= trailingSpacing
;
178 if (leadingSpacing
< 0) {
182 spacing(Primary
, Leading
)->setMinimumWidth(leadingSpacing
);
183 spacing(Primary
, Trailing
)->setFixedWidth(trailingSpacing
);
185 // secondary side of m_splitter
186 if (m_globalXOfSecondary
== INT_MIN
) {
187 Q_ASSERT(m_widthOfSecondary
== INT_MIN
);
190 spacing(Primary
, Trailing
)->setFixedWidth(0);
192 trailingSpacing
= (m_globalXOfSplitter
+ m_splitter
->width())
193 - (m_globalXOfSecondary
+ m_widthOfSecondary
);
194 #if KIO_VERSION < QT_VERSION_CHECK(5, 78, 0)
195 if (trailingSpacing
< 0 || emptyTrashButton(Secondary
)->isVisible()) {
197 if (trailingSpacing
< 0 || emptyTrashButton(Secondary
)->isVisible()
198 || networkFolderButton(Secondary
)->isVisible()
203 const int widthLeftForUrlNavigator2
= m_splitter
->widget(1)->width() - trailingSpacing
;
204 const int widthNeededForUrlNavigator2
= secondaryUrlNavigator()->sizeHint().width() - widthLeftForUrlNavigator2
;
205 if (widthNeededForUrlNavigator2
> 0) {
206 trailingSpacing
-= widthNeededForUrlNavigator2
;
207 if (trailingSpacing
< 0) {
212 spacing(Secondary
, Trailing
)->setMinimumWidth(trailingSpacing
);
215 QWidget
*DolphinNavigatorsWidgetAction::createNavigatorWidget(Side side
) const
217 auto navigatorWidget
= new QWidget(m_splitter
.get());
218 auto layout
= new QHBoxLayout
{navigatorWidget
};
219 layout
->setSpacing(0);
220 layout
->setContentsMargins(0, 0, 0, 0);
221 if (side
== Primary
) {
222 auto leadingSpacing
= new QWidget
{navigatorWidget
};
223 layout
->addWidget(leadingSpacing
);
225 auto urlNavigator
= new DolphinUrlNavigator(navigatorWidget
);
226 layout
->addWidget(urlNavigator
);
228 auto emptyTrashButton
= newEmptyTrashButton(urlNavigator
, navigatorWidget
);
229 layout
->addWidget(emptyTrashButton
);
231 #if !(KIO_VERSION < QT_VERSION_CHECK(5, 78, 0))
232 auto networkFolderButton
= newNetworkFolderButton(urlNavigator
, navigatorWidget
);
233 layout
->addWidget(networkFolderButton
);
236 connect(urlNavigator
, &KUrlNavigator::urlChanged
, this, [this]() {
237 // We have to wait for DolphinUrlNavigator::sizeHint() to update which
238 // happens a little bit later than when urlChanged is emitted.
239 this->m_adjustSpacingTimer
->start();
242 auto trailingSpacing
= new QWidget
{navigatorWidget
};
243 layout
->addWidget(trailingSpacing
);
244 return navigatorWidget
;
247 QPushButton
* DolphinNavigatorsWidgetAction::emptyTrashButton(DolphinNavigatorsWidgetAction::Side side
)
249 int sideIndex
= (side
== Primary
? 0 : 1);
250 if (side
== Primary
) {
251 return static_cast<QPushButton
*>(m_splitter
->widget(sideIndex
)->layout()->itemAt(2)->widget());
253 return static_cast<QPushButton
*>(m_splitter
->widget(sideIndex
)->layout()->itemAt(1)->widget());
256 QPushButton
*DolphinNavigatorsWidgetAction::newEmptyTrashButton(const DolphinUrlNavigator
*urlNavigator
, QWidget
*parent
) const
258 auto emptyTrashButton
= new QPushButton(QIcon::fromTheme(QStringLiteral("user-trash")),
259 i18nc("@action:button", "Empty Trash"), parent
);
260 emptyTrashButton
->setFlat(true);
261 connect(emptyTrashButton
, &QPushButton::clicked
,
262 this, [parent
]() { Trash::empty(parent
); });
263 connect(&Trash::instance(), &Trash::emptinessChanged
,
264 emptyTrashButton
, &QPushButton::setDisabled
);
265 emptyTrashButton
->hide();
266 connect(urlNavigator
, &KUrlNavigator::urlChanged
, this, [emptyTrashButton
, urlNavigator
]() {
267 emptyTrashButton
->setVisible(urlNavigator
->locationUrl().scheme() == QLatin1String("trash"));
269 emptyTrashButton
->setDisabled(Trash::isEmpty());
270 return emptyTrashButton
;
273 #if !(KIO_VERSION < QT_VERSION_CHECK(5, 78, 0))
274 QPushButton
*DolphinNavigatorsWidgetAction::networkFolderButton(DolphinNavigatorsWidgetAction::Side side
)
276 int sideIndex
= (side
== Primary
? 0 : 1);
277 if (side
== Primary
) {
278 return static_cast<QPushButton
*>(m_splitter
->widget(sideIndex
)->layout()->itemAt(3)->widget());
280 return static_cast<QPushButton
*>(m_splitter
->widget(sideIndex
)->layout()->itemAt(2)->widget());
283 QPushButton
*DolphinNavigatorsWidgetAction::newNetworkFolderButton(const DolphinUrlNavigator
*urlNavigator
, QWidget
*parent
) const
285 auto networkFolderButton
= new QPushButton(QIcon::fromTheme(QStringLiteral("folder-add")),
286 i18nc("@action:button", "Add Network Folder"), parent
);
287 networkFolderButton
->setFlat(true);
288 connect(networkFolderButton
, &QPushButton::clicked
,
289 this, [networkFolderButton
]() {
290 KService::Ptr service
= KService::serviceByDesktopName(QStringLiteral("org.kde.knetattach"));
291 auto *job
= new KIO::ApplicationLauncherJob(service
, networkFolderButton
);
292 auto *delegate
= new KNotificationJobUiDelegate
;
293 delegate
->setAutoErrorHandlingEnabled(true);
294 job
->setUiDelegate(delegate
);
297 networkFolderButton
->hide();
298 connect(urlNavigator
, &KUrlNavigator::urlChanged
, this, [networkFolderButton
, urlNavigator
]() {
299 networkFolderButton
->setVisible(urlNavigator
->locationUrl().scheme() == QLatin1String("remote"));
301 return networkFolderButton
;
305 QWidget
*DolphinNavigatorsWidgetAction::spacing(Side side
, Position position
) const
307 int sideIndex
= (side
== Primary
? 0 : 1);
308 if (position
== Leading
) {
309 Q_ASSERT(side
== Primary
); // The secondary side of the splitter has no leading spacing.
310 return m_splitter
->widget(sideIndex
)->layout()->itemAt(0)->widget();
312 if (side
== Primary
) {
313 #if KIO_VERSION < QT_VERSION_CHECK(5, 78, 0)
314 return m_splitter
->widget(sideIndex
)->layout()->itemAt(3)->widget();
316 return m_splitter
->widget(sideIndex
)->layout()->itemAt(4)->widget();
319 #if KIO_VERSION < QT_VERSION_CHECK(5, 78, 0)
320 return m_splitter
->widget(sideIndex
)->layout()->itemAt(2)->widget();
322 return m_splitter
->widget(sideIndex
)->layout()->itemAt(3)->widget();
326 void DolphinNavigatorsWidgetAction::updateText()
328 const int urlNavigatorsAmount
= m_splitter
->count() > 1 && m_splitter
->widget(1)->isVisible() ?
330 setText(i18ncp("@action:inmenu", "Url Navigator", "Url Navigators", urlNavigatorsAmount
));