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>
28 DolphinNavigatorsWidgetAction::DolphinNavigatorsWidgetAction(QWidget
*parent
) :
29 QWidgetAction
{parent
},
30 m_splitter
{new QSplitter(Qt::Horizontal
)},
31 m_adjustSpacingTimer
{new QTimer(this)},
32 m_viewGeometriesHelper
{m_splitter
.get(), this}
35 setIcon(QIcon::fromTheme(QStringLiteral("dialog-scripts")));
37 m_splitter
->setChildrenCollapsible(false);
39 m_splitter
->addWidget(createNavigatorWidget(Primary
));
41 m_adjustSpacingTimer
->setInterval(100);
42 m_adjustSpacingTimer
->setSingleShot(true);
43 connect(m_adjustSpacingTimer
.get(), &QTimer::timeout
,
44 this, &DolphinNavigatorsWidgetAction::adjustSpacing
);
47 void DolphinNavigatorsWidgetAction::adjustSpacing()
49 m_previousWindowWidth
= parentWidget()->window()->width();
50 auto viewGeometries
= m_viewGeometriesHelper
.viewGeometries();
51 const int widthOfSplitterPrimary
= viewGeometries
.globalXOfPrimary
+ viewGeometries
.widthOfPrimary
- viewGeometries
.globalXOfNavigatorsWidget
;
52 const QList
<int> splitterSizes
= {widthOfSplitterPrimary
,
53 m_splitter
->width() - widthOfSplitterPrimary
};
54 m_splitter
->setSizes(splitterSizes
);
56 // primary side of m_splitter
57 int leadingSpacing
= viewGeometries
.globalXOfPrimary
- viewGeometries
.globalXOfNavigatorsWidget
;
58 if (leadingSpacing
< 0) {
61 int trailingSpacing
= (viewGeometries
.globalXOfNavigatorsWidget
+ m_splitter
->width())
62 - (viewGeometries
.globalXOfPrimary
+ viewGeometries
.widthOfPrimary
);
63 if (trailingSpacing
< 0 || emptyTrashButton(Primary
)->isVisible()
64 || networkFolderButton(Primary
)->isVisible()
68 const int widthLeftForUrlNavigator
= m_splitter
->widget(0)->width() - leadingSpacing
- trailingSpacing
;
69 const int widthNeededForUrlNavigator
= primaryUrlNavigator()->sizeHint().width() - widthLeftForUrlNavigator
;
70 if (widthNeededForUrlNavigator
> 0) {
71 trailingSpacing
-= widthNeededForUrlNavigator
;
72 if (trailingSpacing
< 0) {
73 leadingSpacing
+= trailingSpacing
;
76 if (leadingSpacing
< 0) {
80 spacing(Primary
, Leading
)->setMinimumWidth(leadingSpacing
);
81 spacing(Primary
, Trailing
)->setFixedWidth(trailingSpacing
);
83 // secondary side of m_splitter
84 if (viewGeometries
.globalXOfSecondary
== INT_MIN
) {
85 Q_ASSERT(viewGeometries
.widthOfSecondary
== INT_MIN
);
88 spacing(Primary
, Trailing
)->setFixedWidth(0);
90 trailingSpacing
= (viewGeometries
.globalXOfNavigatorsWidget
+ m_splitter
->width())
91 - (viewGeometries
.globalXOfSecondary
+ viewGeometries
.widthOfSecondary
);
92 if (trailingSpacing
< 0 || emptyTrashButton(Secondary
)->isVisible()
93 || networkFolderButton(Secondary
)->isVisible()
97 const int widthLeftForUrlNavigator2
= m_splitter
->widget(1)->width() - trailingSpacing
;
98 const int widthNeededForUrlNavigator2
= secondaryUrlNavigator()->sizeHint().width() - widthLeftForUrlNavigator2
;
99 if (widthNeededForUrlNavigator2
> 0) {
100 trailingSpacing
-= widthNeededForUrlNavigator2
;
101 if (trailingSpacing
< 0) {
106 spacing(Secondary
, Trailing
)->setMinimumWidth(trailingSpacing
);
109 void DolphinNavigatorsWidgetAction::createSecondaryUrlNavigator()
111 Q_ASSERT(m_splitter
->count() == 1);
112 m_splitter
->addWidget(createNavigatorWidget(Secondary
));
113 Q_ASSERT(m_splitter
->count() == 2);
117 void DolphinNavigatorsWidgetAction::followViewContainersGeometry(QWidget
*primaryViewContainer
,
118 QWidget
*secondaryViewContainer
)
120 m_viewGeometriesHelper
.setViewContainers(primaryViewContainer
, secondaryViewContainer
);
124 bool DolphinNavigatorsWidgetAction::isInToolbar() const
126 return qobject_cast
<QToolBar
*>(m_splitter
->parentWidget());
129 DolphinUrlNavigator
* DolphinNavigatorsWidgetAction::primaryUrlNavigator() const
131 Q_ASSERT(m_splitter
);
132 return m_splitter
->widget(0)->findChild
<DolphinUrlNavigator
*>();
135 DolphinUrlNavigator
* DolphinNavigatorsWidgetAction::secondaryUrlNavigator() const
137 Q_ASSERT(m_splitter
);
138 if (m_splitter
->count() < 2) {
141 return m_splitter
->widget(1)->findChild
<DolphinUrlNavigator
*>();
144 void DolphinNavigatorsWidgetAction::setSecondaryNavigatorVisible(bool visible
)
147 Q_ASSERT(m_splitter
->count() == 2);
148 m_splitter
->widget(1)->setVisible(true);
149 } else if (m_splitter
->count() > 1) {
150 m_splitter
->widget(1)->setVisible(false);
151 // Fix an unlikely event of wrong trash button visibility.
152 emptyTrashButton(Secondary
)->setVisible(false);
157 QWidget
*DolphinNavigatorsWidgetAction::createWidget(QWidget
*parent
)
159 QWidget
*oldParent
= m_splitter
->parentWidget();
160 if (oldParent
&& oldParent
->layout()) {
161 oldParent
->layout()->removeWidget(m_splitter
.get());
162 QGridLayout
*layout
= qobject_cast
<QGridLayout
*>(oldParent
->layout());
163 if (qobject_cast
<QToolBar
*>(parent
) && layout
) {
164 // in DolphinTabPage::insertNavigatorsWidget the minimumHeight of this row was
165 // set to fit the m_splitter. Since we are now removing it again, the
166 // minimumHeight can be reset to 0.
167 layout
->setRowMinimumHeight(0, 0);
170 m_splitter
->setParent(parent
);
171 return m_splitter
.get();
174 void DolphinNavigatorsWidgetAction::deleteWidget(QWidget
*widget
)
177 m_splitter
->setParent(nullptr);
180 QWidget
*DolphinNavigatorsWidgetAction::createNavigatorWidget(Side side
) const
182 auto navigatorWidget
= new QWidget(m_splitter
.get());
183 auto layout
= new QHBoxLayout
{navigatorWidget
};
184 layout
->setSpacing(0);
185 layout
->setContentsMargins(0, 0, 0, 0);
186 if (side
== Primary
) {
187 auto leadingSpacing
= new QWidget
{navigatorWidget
};
188 layout
->addWidget(leadingSpacing
);
190 auto urlNavigator
= new DolphinUrlNavigator(navigatorWidget
);
191 layout
->addWidget(urlNavigator
);
193 auto emptyTrashButton
= newEmptyTrashButton(urlNavigator
, navigatorWidget
);
194 layout
->addWidget(emptyTrashButton
);
196 auto networkFolderButton
= newNetworkFolderButton(urlNavigator
, navigatorWidget
);
197 layout
->addWidget(networkFolderButton
);
199 connect(urlNavigator
, &KUrlNavigator::urlChanged
, this, [urlNavigator
, this]() {
200 // Update URL navigator to show a server URL entry placeholder text if we
201 // just loaded the remote:/ page, to make it easier for users to figure out
202 // that they can enter arbitrary remote URLs. See bug 414670
203 if (urlNavigator
->locationUrl().scheme() == QLatin1String("remote")) {
204 if (!urlNavigator
->isUrlEditable()) {
205 urlNavigator
->setUrlEditable(true);
207 urlNavigator
->clearText();
208 urlNavigator
->setPlaceholderText(i18n("Enter server URL (e.g. smb://[ip address])"));
210 urlNavigator
->setPlaceholderText(QString());
213 // We have to wait for DolphinUrlNavigator::sizeHint() to update which
214 // happens a little bit later than when urlChanged is emitted.
215 this->m_adjustSpacingTimer
->start();
216 }, Qt::QueuedConnection
);
218 auto trailingSpacing
= new QWidget
{navigatorWidget
};
219 layout
->addWidget(trailingSpacing
);
220 return navigatorWidget
;
223 QPushButton
* DolphinNavigatorsWidgetAction::emptyTrashButton(DolphinNavigatorsWidgetAction::Side side
)
225 int sideIndex
= (side
== Primary
? 0 : 1);
226 if (side
== Primary
) {
227 return static_cast<QPushButton
*>(m_splitter
->widget(sideIndex
)->layout()->itemAt(2)->widget());
229 return static_cast<QPushButton
*>(m_splitter
->widget(sideIndex
)->layout()->itemAt(1)->widget());
232 QPushButton
*DolphinNavigatorsWidgetAction::newEmptyTrashButton(const DolphinUrlNavigator
*urlNavigator
, QWidget
*parent
) const
234 auto emptyTrashButton
= new QPushButton(QIcon::fromTheme(QStringLiteral("user-trash")),
235 i18nc("@action:button", "Empty Trash"), parent
);
236 emptyTrashButton
->setToolTip(i18n("Empties Trash to create free space"));
238 emptyTrashButton
->setFlat(true);
239 connect(emptyTrashButton
, &QPushButton::clicked
,
240 this, [parent
]() { Trash::empty(parent
); });
241 connect(&Trash::instance(), &Trash::emptinessChanged
,
242 emptyTrashButton
, &QPushButton::setDisabled
);
243 emptyTrashButton
->hide();
244 connect(urlNavigator
, &KUrlNavigator::urlChanged
, this, [emptyTrashButton
, urlNavigator
]() {
245 emptyTrashButton
->setVisible(urlNavigator
->locationUrl().scheme() == QLatin1String("trash"));
247 emptyTrashButton
->setDisabled(Trash::isEmpty());
248 return emptyTrashButton
;
251 QPushButton
*DolphinNavigatorsWidgetAction::networkFolderButton(DolphinNavigatorsWidgetAction::Side side
)
253 int sideIndex
= (side
== Primary
? 0 : 1);
254 if (side
== Primary
) {
255 return static_cast<QPushButton
*>(m_splitter
->widget(sideIndex
)->layout()->itemAt(3)->widget());
257 return static_cast<QPushButton
*>(m_splitter
->widget(sideIndex
)->layout()->itemAt(2)->widget());
260 QPushButton
*DolphinNavigatorsWidgetAction::newNetworkFolderButton(const DolphinUrlNavigator
*urlNavigator
, QWidget
*parent
) const
262 auto networkFolderButton
= new QPushButton(QIcon::fromTheme(QStringLiteral("folder-add")),
263 i18nc("@action:button", "Add Network Folder"), parent
);
264 networkFolderButton
->setFlat(true);
265 KService::Ptr service
= KService::serviceByDesktopName(QStringLiteral("org.kde.knetattach"));
266 connect(networkFolderButton
, &QPushButton::clicked
,
267 this, [networkFolderButton
, service
]() {
268 auto *job
= new KIO::ApplicationLauncherJob(service
, networkFolderButton
);
269 auto *delegate
= new KNotificationJobUiDelegate
;
270 delegate
->setAutoErrorHandlingEnabled(true);
271 job
->setUiDelegate(delegate
);
274 networkFolderButton
->hide();
275 connect(urlNavigator
, &KUrlNavigator::urlChanged
, this, [networkFolderButton
, urlNavigator
, service
]() {
276 networkFolderButton
->setVisible(service
&& urlNavigator
->locationUrl().scheme() == QLatin1String("remote"));
278 return networkFolderButton
;
281 QWidget
*DolphinNavigatorsWidgetAction::spacing(Side side
, Position position
) const
283 int sideIndex
= (side
== Primary
? 0 : 1);
284 if (position
== Leading
) {
285 Q_ASSERT(side
== Primary
); // The secondary side of the splitter has no leading spacing.
286 return m_splitter
->widget(sideIndex
)->layout()->itemAt(0)->widget();
288 if (side
== Primary
) {
289 return m_splitter
->widget(sideIndex
)->layout()->itemAt(4)->widget();
291 return m_splitter
->widget(sideIndex
)->layout()->itemAt(3)->widget();
294 void DolphinNavigatorsWidgetAction::updateText()
296 const int urlNavigatorsAmount
= m_splitter
->count() > 1 && m_splitter
->widget(1)->isVisible() ?
298 setText(i18ncp("@action:inmenu", "Location Bar", "Location Bars", urlNavigatorsAmount
));
301 DolphinNavigatorsWidgetAction::ViewGeometriesHelper::ViewGeometriesHelper
302 (QWidget
*navigatorsWidget
, DolphinNavigatorsWidgetAction
*navigatorsWidgetAction
) :
303 m_navigatorsWidget
{navigatorsWidget
},
304 m_navigatorsWidgetAction
{navigatorsWidgetAction
}
306 Q_CHECK_PTR(navigatorsWidget
);
307 Q_CHECK_PTR(navigatorsWidgetAction
);
310 bool DolphinNavigatorsWidgetAction::ViewGeometriesHelper::eventFilter(QObject
*watched
, QEvent
*event
)
312 if (event
->type() == QEvent::Resize
) {
313 if (m_navigatorsWidgetAction
->parentWidget()->window()->width() != m_navigatorsWidgetAction
->m_previousWindowWidth
) {
314 // The window is being resized which means not all widgets have gotten their new sizes yet.
315 // Let's wait a bit so the sizes of the navigatorsWidget and the viewContainers have all
316 // had a chance to be updated.
317 m_navigatorsWidgetAction
->m_adjustSpacingTimer
->start();
319 m_navigatorsWidgetAction
->adjustSpacing();
320 // We could always use the m_adjustSpacingTimer instead of calling adjustSpacing() directly
321 // here but then the navigatorsWidget doesn't fluently align with the viewContainers when
322 // the DolphinTabPage::m_expandViewAnimation is animating.
326 return QObject::eventFilter(watched
, event
);
329 void DolphinNavigatorsWidgetAction::ViewGeometriesHelper::setViewContainers(QWidget
*primaryViewContainer
,
330 QWidget
*secondaryViewContainer
)
332 Q_CHECK_PTR(primaryViewContainer
);
333 if (m_primaryViewContainer
) {
334 m_primaryViewContainer
->removeEventFilter(this);
336 primaryViewContainer
->installEventFilter(this);
337 m_primaryViewContainer
= primaryViewContainer
;
339 // It is not possible to resize the secondaryViewContainer without simultaneously
340 // resizing the primaryViewContainer so we don't have to installEventFilter() here.
341 m_secondaryViewContainer
= secondaryViewContainer
;
344 DolphinNavigatorsWidgetAction::ViewGeometriesHelper::Geometries
345 DolphinNavigatorsWidgetAction::ViewGeometriesHelper::viewGeometries()
347 Q_ASSERT(m_primaryViewContainer
);
348 Geometries geometries
;
351 geometries
.widthOfPrimary
= m_primaryViewContainer
->width();
352 if (m_secondaryViewContainer
) {
353 geometries
.widthOfSecondary
= m_secondaryViewContainer
->width();
355 geometries
.widthOfSecondary
= INT_MIN
;
359 if (QApplication::layoutDirection() == Qt::LeftToRight
) {
360 geometries
.globalXOfNavigatorsWidget
= m_navigatorsWidget
->mapToGlobal(QPoint(0,0)).x();
361 geometries
.globalXOfPrimary
= m_primaryViewContainer
->mapToGlobal(QPoint(0,0)).x();
362 geometries
.globalXOfSecondary
= !m_secondaryViewContainer
? INT_MIN
:
363 m_secondaryViewContainer
->mapToGlobal(QPoint(0,0)).x();
365 // When the direction is reversed, globalX does not change.
366 // For the adjustSpacing() code to work we need globalX to measure from right to left
367 // and to measure up to the rightmost point of a widget instead of the leftmost.
368 geometries
.globalXOfNavigatorsWidget
=
369 (-1) * (m_navigatorsWidget
->mapToGlobal(QPoint(0,0)).x() + m_navigatorsWidget
->width());
370 geometries
.globalXOfPrimary
=
371 (-1) * (m_primaryViewContainer
->mapToGlobal(QPoint(0,0)).x() + geometries
.widthOfPrimary
);
372 geometries
.globalXOfSecondary
= !m_secondaryViewContainer
? INT_MIN
:
373 (-1) * (m_secondaryViewContainer
->mapToGlobal(QPoint(0,0)).x() + geometries
.widthOfSecondary
);