2 This file is part of the KDE project
3 SPDX-FileCopyrightText: 2020 Felix Ernst <felixernst@kde.org>
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 <KCoreAddons>
13 #include <KLocalizedString>
14 #include <KNotificationJobUiDelegate>
17 #include <KIO/ApplicationLauncherJob>
19 #include <QApplication>
20 #include <QHBoxLayout>
21 #include <QPushButton>
27 DolphinNavigatorsWidgetAction::DolphinNavigatorsWidgetAction(QWidget
*parent
)
28 : QWidgetAction
{parent
}
29 , m_splitter
{new QSplitter(Qt::Horizontal
)}
30 , m_adjustSpacingTimer
{new QTimer(this)}
31 , m_viewGeometriesHelper
{m_splitter
.get(), this}
34 setIcon(QIcon::fromTheme(QStringLiteral("dialog-scripts")));
36 m_splitter
->setChildrenCollapsible(false);
38 m_splitter
->addWidget(createNavigatorWidget(Primary
));
39 m_splitter
->setFocusProxy(primaryUrlNavigator());
41 m_adjustSpacingTimer
->setInterval(100);
42 m_adjustSpacingTimer
->setSingleShot(true);
43 connect(m_adjustSpacingTimer
.get(), &QTimer::timeout
, this, &DolphinNavigatorsWidgetAction::adjustSpacing
);
46 void DolphinNavigatorsWidgetAction::adjustSpacing()
48 m_previousWindowWidth
= qobject_cast
<QWidget
*>(parent())->window()->width();
49 auto viewGeometries
= m_viewGeometriesHelper
.viewGeometries();
50 const int widthOfSplitterPrimary
= viewGeometries
.globalXOfPrimary
+ viewGeometries
.widthOfPrimary
- viewGeometries
.globalXOfNavigatorsWidget
;
51 const QList
<int> splitterSizes
= {widthOfSplitterPrimary
, m_splitter
->width() - widthOfSplitterPrimary
- m_splitter
->handleWidth()};
52 m_splitter
->setSizes(splitterSizes
);
54 // primary side of m_splitter
55 int leadingSpacing
= viewGeometries
.globalXOfPrimary
- viewGeometries
.globalXOfNavigatorsWidget
;
56 if (leadingSpacing
< 0) {
59 int trailingSpacing
= (viewGeometries
.globalXOfNavigatorsWidget
+ m_splitter
->width()) - (viewGeometries
.globalXOfPrimary
+ viewGeometries
.widthOfPrimary
);
60 if (trailingSpacing
< 0 || emptyTrashButton(Primary
)->isVisible() || networkFolderButton(Primary
)->isVisible()) {
63 const int widthLeftForUrlNavigator
= m_splitter
->widget(0)->width() - leadingSpacing
- trailingSpacing
;
64 const int widthNeededForUrlNavigator
= primaryUrlNavigator()->sizeHint().width() - widthLeftForUrlNavigator
;
65 if (widthNeededForUrlNavigator
> 0) {
66 trailingSpacing
-= widthNeededForUrlNavigator
;
67 if (trailingSpacing
< 0) {
68 leadingSpacing
+= trailingSpacing
;
71 if (leadingSpacing
< 0) {
75 spacing(Primary
, Leading
)->setMinimumWidth(leadingSpacing
);
76 spacing(Primary
, Trailing
)->setFixedWidth(trailingSpacing
);
78 // secondary side of m_splitter
79 if (viewGeometries
.globalXOfSecondary
== INT_MIN
) {
80 Q_ASSERT(viewGeometries
.widthOfSecondary
== INT_MIN
);
83 spacing(Primary
, Trailing
)->setFixedWidth(0);
85 trailingSpacing
= (viewGeometries
.globalXOfNavigatorsWidget
+ m_splitter
->width()) - (viewGeometries
.globalXOfSecondary
+ viewGeometries
.widthOfSecondary
);
86 if (trailingSpacing
< 0 || emptyTrashButton(Secondary
)->isVisible() || networkFolderButton(Secondary
)->isVisible()) {
89 const int widthLeftForUrlNavigator2
= m_splitter
->widget(1)->width() - trailingSpacing
;
90 const int widthNeededForUrlNavigator2
= secondaryUrlNavigator()->sizeHint().width() - widthLeftForUrlNavigator2
;
91 if (widthNeededForUrlNavigator2
> 0) {
92 trailingSpacing
-= widthNeededForUrlNavigator2
;
93 if (trailingSpacing
< 0) {
98 spacing(Secondary
, Trailing
)->setMinimumWidth(trailingSpacing
);
101 void DolphinNavigatorsWidgetAction::createSecondaryUrlNavigator()
103 Q_ASSERT(m_splitter
->count() == 1);
104 m_splitter
->addWidget(createNavigatorWidget(Secondary
));
105 Q_ASSERT(m_splitter
->count() == 2);
106 #if KIO_VERSION >= QT_VERSION_CHECK(6, 14, 0)
107 secondaryUrlNavigator()->setBackgroundEnabled(primaryUrlNavigator()->isBackgroundEnabled());
112 void DolphinNavigatorsWidgetAction::followViewContainersGeometry(QWidget
*primaryViewContainer
, QWidget
*secondaryViewContainer
)
114 m_viewGeometriesHelper
.setViewContainers(primaryViewContainer
, secondaryViewContainer
);
118 bool DolphinNavigatorsWidgetAction::isInToolbar() const
120 return qobject_cast
<QToolBar
*>(m_splitter
->parentWidget());
123 DolphinUrlNavigator
*DolphinNavigatorsWidgetAction::primaryUrlNavigator() const
125 Q_ASSERT(m_splitter
);
126 return m_splitter
->widget(0)->findChild
<DolphinUrlNavigator
*>();
129 DolphinUrlNavigator
*DolphinNavigatorsWidgetAction::secondaryUrlNavigator() const
131 Q_ASSERT(m_splitter
);
132 if (m_splitter
->count() < 2) {
135 return m_splitter
->widget(1)->findChild
<DolphinUrlNavigator
*>();
138 void DolphinNavigatorsWidgetAction::setSecondaryNavigatorVisible(bool visible
)
141 Q_ASSERT(m_splitter
->count() == 2);
142 m_splitter
->widget(0)->setContentsMargins(0, 0, m_splitter
->style()->pixelMetric(QStyle::PM_LayoutRightMargin
), 0);
143 m_splitter
->widget(1)->setContentsMargins(m_splitter
->style()->pixelMetric(QStyle::PM_LayoutLeftMargin
), 0, 0, 0);
144 m_splitter
->widget(1)->setVisible(true);
145 } else if (m_splitter
->count() > 1) {
146 m_splitter
->widget(1)->setVisible(false);
147 m_splitter
->widget(0)->setContentsMargins(0, 0, 0, 0);
148 m_splitter
->widget(1)->setContentsMargins(0, 0, 0, 0);
149 // Fix an unlikely event of wrong trash button visibility.
150 emptyTrashButton(Secondary
)->setVisible(false);
155 void DolphinNavigatorsWidgetAction::setBackgroundEnabled(bool enabled
)
157 #if KIO_VERSION >= QT_VERSION_CHECK(6, 14, 0)
158 m_splitter
->setAutoFillBackground(!enabled
);
159 m_splitter
->setBackgroundRole(enabled
? QPalette::Window
: QPalette::Base
);
160 primaryUrlNavigator()->setBackgroundEnabled(enabled
);
161 if (secondaryUrlNavigator()) {
162 secondaryUrlNavigator()->setBackgroundEnabled(enabled
);
169 QWidget
*DolphinNavigatorsWidgetAction::createWidget(QWidget
*parent
)
171 QWidget
*oldParent
= m_splitter
->parentWidget();
172 if (oldParent
&& oldParent
->layout()) {
173 oldParent
->layout()->removeWidget(m_splitter
.get());
174 QGridLayout
*layout
= qobject_cast
<QGridLayout
*>(oldParent
->layout());
175 if (qobject_cast
<QToolBar
*>(parent
) && layout
) {
176 // in DolphinTabPage::insertNavigatorsWidget the minimumHeight of this row was
177 // set to fit the m_splitter. Since we are now removing it again, the
178 // minimumHeight can be reset to 0.
179 layout
->setRowMinimumHeight(0, 0);
182 m_splitter
->setParent(parent
);
183 return m_splitter
.get();
186 void DolphinNavigatorsWidgetAction::deleteWidget(QWidget
*widget
)
189 m_splitter
->setParent(nullptr);
192 QWidget
*DolphinNavigatorsWidgetAction::createNavigatorWidget(Side side
) const
194 auto navigatorWidget
= new QWidget(m_splitter
.get());
195 auto layout
= new QHBoxLayout
{navigatorWidget
};
196 layout
->setSpacing(0);
197 layout
->setContentsMargins(0, 0, 0, 0);
199 if (side
== Primary
) {
200 auto leadingSpacing
= new QWidget
{navigatorWidget
};
201 layout
->addWidget(leadingSpacing
);
203 auto urlNavigator
= new DolphinUrlNavigator(navigatorWidget
);
204 layout
->addWidget(urlNavigator
);
206 auto emptyTrashButton
= newEmptyTrashButton(urlNavigator
, navigatorWidget
);
207 layout
->addWidget(emptyTrashButton
);
209 auto networkFolderButton
= newNetworkFolderButton(urlNavigator
, navigatorWidget
);
210 layout
->addWidget(networkFolderButton
);
214 &KUrlNavigator::urlChanged
,
216 [urlNavigator
, this]() {
217 // Update URL navigator to show a server URL entry placeholder text if we
218 // just loaded the remote:/ page, to make it easier for users to figure out
219 // that they can enter arbitrary remote URLs. See bug 414670
220 if (urlNavigator
->locationUrl().scheme() == QLatin1String("remote")) {
221 if (!urlNavigator
->isUrlEditable()) {
222 urlNavigator
->setUrlEditable(true);
224 urlNavigator
->clearText();
225 urlNavigator
->setPlaceholderText(i18n("Enter server URL (e.g. smb://[ip address])"));
227 urlNavigator
->setPlaceholderText(QString());
230 // We have to wait for DolphinUrlNavigator::sizeHint() to update which
231 // happens a little bit later than when urlChanged is emitted.
232 this->m_adjustSpacingTimer
->start();
234 Qt::QueuedConnection
);
236 auto trailingSpacing
= new QWidget
{navigatorWidget
};
237 layout
->addWidget(trailingSpacing
);
238 return navigatorWidget
;
241 QPushButton
*DolphinNavigatorsWidgetAction::emptyTrashButton(DolphinNavigatorsWidgetAction::Side side
)
243 int sideIndex
= (side
== Primary
? 0 : 1);
244 if (side
== Primary
) {
245 return static_cast<QPushButton
*>(m_splitter
->widget(sideIndex
)->layout()->itemAt(2)->widget());
247 return static_cast<QPushButton
*>(m_splitter
->widget(sideIndex
)->layout()->itemAt(1)->widget());
250 QPushButton
*DolphinNavigatorsWidgetAction::newEmptyTrashButton(const DolphinUrlNavigator
*urlNavigator
, QWidget
*parent
) const
252 auto emptyTrashButton
= new QPushButton(QIcon::fromTheme(QStringLiteral("edit-delete")), i18nc("@action:button", "Empty Trash"), parent
);
253 emptyTrashButton
->setToolTip(i18n("Empties Trash to create free space"));
255 emptyTrashButton
->setFlat(true);
256 connect(emptyTrashButton
, &QPushButton::clicked
, this, [parent
]() {
257 Trash::empty(parent
);
259 connect(&Trash::instance(), &Trash::emptinessChanged
, emptyTrashButton
, &QPushButton::setDisabled
);
260 emptyTrashButton
->hide();
261 connect(urlNavigator
, &KUrlNavigator::urlChanged
, this, [emptyTrashButton
, urlNavigator
]() {
262 emptyTrashButton
->setVisible(urlNavigator
->locationUrl().scheme() == QLatin1String("trash"));
264 emptyTrashButton
->setDisabled(Trash::isEmpty());
265 return emptyTrashButton
;
268 QPushButton
*DolphinNavigatorsWidgetAction::networkFolderButton(DolphinNavigatorsWidgetAction::Side side
)
270 int sideIndex
= (side
== Primary
? 0 : 1);
271 if (side
== Primary
) {
272 return static_cast<QPushButton
*>(m_splitter
->widget(sideIndex
)->layout()->itemAt(3)->widget());
274 return static_cast<QPushButton
*>(m_splitter
->widget(sideIndex
)->layout()->itemAt(2)->widget());
277 QPushButton
*DolphinNavigatorsWidgetAction::newNetworkFolderButton(const DolphinUrlNavigator
*urlNavigator
, QWidget
*parent
) const
279 auto networkFolderButton
= new QPushButton(QIcon::fromTheme(QStringLiteral("folder-add")), i18nc("@action:button", "Add Network Folder"), parent
);
280 networkFolderButton
->setFlat(true);
281 connect(networkFolderButton
, &QPushButton::clicked
, this, [networkFolderButton
]() {
282 const KService::Ptr service
= KService::serviceByDesktopName(QStringLiteral("org.kde.knetattach"));
283 auto *job
= new KIO::ApplicationLauncherJob(service
, networkFolderButton
);
284 auto *delegate
= new KNotificationJobUiDelegate
;
285 delegate
->setAutoErrorHandlingEnabled(true);
286 job
->setUiDelegate(delegate
);
289 networkFolderButton
->hide();
290 connect(urlNavigator
, &KUrlNavigator::urlChanged
, this, [networkFolderButton
, urlNavigator
]() {
291 if (urlNavigator
->locationUrl().scheme() == QLatin1String("remote")) {
292 // Looking up a service can be a bit slow, so we only do it now when it becomes necessary.
293 const KService::Ptr service
= KService::serviceByDesktopName(QStringLiteral("org.kde.knetattach"));
294 networkFolderButton
->setVisible(service
);
296 networkFolderButton
->setVisible(false);
299 return networkFolderButton
;
302 QWidget
*DolphinNavigatorsWidgetAction::spacing(Side side
, Position position
) const
304 int sideIndex
= (side
== Primary
? 0 : 1);
305 if (position
== Leading
) {
306 Q_ASSERT(side
== Primary
); // The secondary side of the splitter has no leading spacing.
307 return m_splitter
->widget(sideIndex
)->layout()->itemAt(0)->widget();
309 if (side
== Primary
) {
310 return m_splitter
->widget(sideIndex
)->layout()->itemAt(4)->widget();
312 return m_splitter
->widget(sideIndex
)->layout()->itemAt(3)->widget();
315 void DolphinNavigatorsWidgetAction::updateText()
317 const int urlNavigatorsAmount
= m_splitter
->count() > 1 && m_splitter
->widget(1)->isVisible() ? 2 : 1;
318 setText(i18ncp("@action:inmenu", "Location Bar", "Location Bars", urlNavigatorsAmount
));
321 DolphinNavigatorsWidgetAction::ViewGeometriesHelper::ViewGeometriesHelper(QWidget
*navigatorsWidget
, DolphinNavigatorsWidgetAction
*navigatorsWidgetAction
)
322 : m_navigatorsWidget
{navigatorsWidget
}
323 , m_navigatorsWidgetAction
{navigatorsWidgetAction
}
325 Q_CHECK_PTR(navigatorsWidget
);
326 Q_CHECK_PTR(navigatorsWidgetAction
);
329 bool DolphinNavigatorsWidgetAction::ViewGeometriesHelper::eventFilter(QObject
*watched
, QEvent
*event
)
331 if (event
->type() == QEvent::Resize
) {
332 if (qobject_cast
<QWidget
*>(m_navigatorsWidgetAction
->parent())->window()->width() != m_navigatorsWidgetAction
->m_previousWindowWidth
) {
333 // The window is being resized which means not all widgets have gotten their new sizes yet.
334 // Let's wait a bit so the sizes of the navigatorsWidget and the viewContainers have all
335 // had a chance to be updated.
336 m_navigatorsWidgetAction
->m_adjustSpacingTimer
->start();
338 m_navigatorsWidgetAction
->adjustSpacing();
339 // We could always use the m_adjustSpacingTimer instead of calling adjustSpacing() directly
340 // here but then the navigatorsWidget doesn't fluently align with the viewContainers when
341 // the DolphinTabPage::m_expandViewAnimation is animating.
345 return QObject::eventFilter(watched
, event
);
348 void DolphinNavigatorsWidgetAction::ViewGeometriesHelper::setViewContainers(QWidget
*primaryViewContainer
, QWidget
*secondaryViewContainer
)
350 Q_CHECK_PTR(primaryViewContainer
);
351 if (m_primaryViewContainer
) {
352 m_primaryViewContainer
->removeEventFilter(this);
354 primaryViewContainer
->installEventFilter(this);
355 m_primaryViewContainer
= primaryViewContainer
;
357 // It is not possible to resize the secondaryViewContainer without simultaneously
358 // resizing the primaryViewContainer so we don't have to installEventFilter() here.
359 m_secondaryViewContainer
= secondaryViewContainer
;
362 DolphinNavigatorsWidgetAction::ViewGeometriesHelper::Geometries
DolphinNavigatorsWidgetAction::ViewGeometriesHelper::viewGeometries()
364 Q_ASSERT(m_primaryViewContainer
);
365 Geometries geometries
;
368 geometries
.widthOfPrimary
= m_primaryViewContainer
->width();
369 if (m_secondaryViewContainer
) {
370 geometries
.widthOfSecondary
= m_secondaryViewContainer
->width();
372 geometries
.widthOfSecondary
= INT_MIN
;
376 if (QApplication::layoutDirection() == Qt::LeftToRight
) {
377 geometries
.globalXOfNavigatorsWidget
= m_navigatorsWidget
->mapToGlobal(QPoint(0, 0)).x();
378 geometries
.globalXOfPrimary
= m_primaryViewContainer
->mapToGlobal(QPoint(0, 0)).x();
379 geometries
.globalXOfSecondary
= !m_secondaryViewContainer
? INT_MIN
: m_secondaryViewContainer
->mapToGlobal(QPoint(0, 0)).x();
381 // When the direction is reversed, globalX does not change.
382 // For the adjustSpacing() code to work we need globalX to measure from right to left
383 // and to measure up to the rightmost point of a widget instead of the leftmost.
384 geometries
.globalXOfNavigatorsWidget
= (-1) * (m_navigatorsWidget
->mapToGlobal(QPoint(0, 0)).x() + m_navigatorsWidget
->width());
385 geometries
.globalXOfPrimary
= (-1) * (m_primaryViewContainer
->mapToGlobal(QPoint(0, 0)).x() + geometries
.widthOfPrimary
);
386 geometries
.globalXOfSecondary
=
387 !m_secondaryViewContainer
? INT_MIN
: (-1) * (m_secondaryViewContainer
->mapToGlobal(QPoint(0, 0)).x() + geometries
.widthOfSecondary
);
392 #include "moc_dolphinnavigatorswidgetaction.cpp"