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 <KXMLGUIFactory>
14 #include <KXmlGuiWindow>
16 #include <QApplication>
17 #include <QDomDocument>
18 #include <QHBoxLayout>
19 #include <QPushButton>
24 DolphinNavigatorsWidgetAction::DolphinNavigatorsWidgetAction(QWidget
*parent
) :
25 QWidgetAction
{parent
},
26 m_splitter
{new QSplitter(Qt::Horizontal
)},
27 m_adjustSpacingTimer
{new QTimer(this)},
28 m_globalXOfSplitter
{INT_MIN
},
29 m_globalXOfPrimary
{INT_MIN
},
30 m_widthOfPrimary
{INT_MIN
},
31 m_globalXOfSecondary
{INT_MIN
},
32 m_widthOfSecondary
{INT_MIN
}
35 setIcon(QIcon::fromTheme(QStringLiteral("dialog-scripts")));
37 m_splitter
->setChildrenCollapsible(false);
38 setDefaultWidget(m_splitter
.get());
40 m_splitter
->addWidget(createNavigatorWidget(Primary
));
42 m_adjustSpacingTimer
->setInterval(100);
43 m_adjustSpacingTimer
->setSingleShot(true);
44 connect(m_adjustSpacingTimer
.get(), &QTimer::timeout
,
45 this, &DolphinNavigatorsWidgetAction::adjustSpacing
);
48 bool DolphinNavigatorsWidgetAction::addToToolbarAndSave(KXmlGuiWindow
*mainWindow
)
50 const QString rawXml
= KXMLGUIFactory::readConfigFile(mainWindow
->xmlFile());
51 QDomDocument domDocument
;
52 if (rawXml
.isEmpty() || !domDocument
.setContent(rawXml
) || domDocument
.isNull()) {
55 QDomNode toolbar
= domDocument
.elementsByTagName(QStringLiteral("ToolBar")).at(0);
56 if (toolbar
.isNull()) {
60 QDomElement urlNavigatorElement
= domDocument
.createElement(QStringLiteral("Action"));
61 urlNavigatorElement
.setAttribute(QStringLiteral("name"), QStringLiteral("url_navigators"));
63 QDomNode position
= toolbar
.firstChildElement(QStringLiteral("Spacer"));
64 if (position
.isNull()) {
65 toolbar
.appendChild(urlNavigatorElement
);
67 toolbar
.replaceChild(urlNavigatorElement
, position
);
70 KXMLGUIFactory::saveConfigFile(domDocument
, mainWindow
->xmlFile());
71 mainWindow
->reloadXML();
72 mainWindow
->createGUI();
76 void DolphinNavigatorsWidgetAction::createSecondaryUrlNavigator()
78 Q_ASSERT(m_splitter
->count() == 1);
79 m_splitter
->addWidget(createNavigatorWidget(Secondary
));
80 Q_ASSERT(m_splitter
->count() == 2);
84 void DolphinNavigatorsWidgetAction::followViewContainerGeometry(
85 int globalXOfPrimary
, int widthOfPrimary
)
87 followViewContainersGeometry(globalXOfPrimary
, widthOfPrimary
, INT_MIN
, INT_MIN
);
90 void DolphinNavigatorsWidgetAction::followViewContainersGeometry(
91 int globalXOfPrimary
, int widthOfPrimary
,
92 int globalXOfSecondary
, int widthOfSecondary
)
94 if (QApplication::layoutDirection() == Qt::LeftToRight
) {
95 m_globalXOfSplitter
= m_splitter
->mapToGlobal(QPoint(0,0)).x();
96 m_globalXOfPrimary
= globalXOfPrimary
;
97 m_globalXOfSecondary
= globalXOfSecondary
;
99 // When the direction is reversed, globalX does not change.
100 // For the adjustSpacing() code to work we need globalX to measure from right to left
101 // and to measure up to the rightmost point of a widget instead of the leftmost.
102 m_globalXOfSplitter
= (-1) * (m_splitter
->mapToGlobal(QPoint(0,0)).x() + m_splitter
->width());
103 m_globalXOfPrimary
= (-1) * (globalXOfPrimary
+ widthOfPrimary
);
104 m_globalXOfSecondary
= (globalXOfSecondary
== INT_MIN
) ? INT_MIN
:
105 (-1) * (globalXOfSecondary
+ widthOfSecondary
);
107 m_widthOfPrimary
= widthOfPrimary
;
108 m_widthOfSecondary
= widthOfSecondary
;
112 DolphinUrlNavigator
* DolphinNavigatorsWidgetAction::primaryUrlNavigator() const
114 Q_ASSERT(m_splitter
);
115 return m_splitter
->widget(0)->findChild
<DolphinUrlNavigator
*>();
118 DolphinUrlNavigator
* DolphinNavigatorsWidgetAction::secondaryUrlNavigator() const
120 Q_ASSERT(m_splitter
);
121 if (m_splitter
->count() < 2) {
124 return m_splitter
->widget(1)->findChild
<DolphinUrlNavigator
*>();
127 void DolphinNavigatorsWidgetAction::setSecondaryNavigatorVisible(bool visible
)
130 Q_ASSERT(m_splitter
->count() == 2);
131 m_splitter
->widget(1)->setVisible(true);
132 } else if (m_splitter
->count() > 1) {
133 m_splitter
->widget(1)->setVisible(false);
134 // Fix an unlikely event of wrong trash button visibility.
135 emptyTrashButton(Secondary
)->setVisible(false);
140 void DolphinNavigatorsWidgetAction::adjustSpacing()
142 Q_ASSERT(m_globalXOfSplitter
!= INT_MIN
);
143 Q_ASSERT(m_globalXOfPrimary
!= INT_MIN
);
144 Q_ASSERT(m_widthOfPrimary
!= INT_MIN
);
145 const int widthOfSplitterPrimary
= m_globalXOfPrimary
+ m_widthOfPrimary
- m_globalXOfSplitter
;
146 const QList
<int> splitterSizes
= {widthOfSplitterPrimary
,
147 m_splitter
->width() - widthOfSplitterPrimary
};
148 m_splitter
->setSizes(splitterSizes
);
150 // primary side of m_splitter
151 int leadingSpacing
= m_globalXOfPrimary
- m_globalXOfSplitter
;
152 if (leadingSpacing
< 0) {
155 int trailingSpacing
= (m_globalXOfSplitter
+ m_splitter
->width())
156 - (m_globalXOfPrimary
+ m_widthOfPrimary
);
157 if (trailingSpacing
< 0 || emptyTrashButton(Primary
)->isVisible()) {
160 const int widthLeftForUrlNavigator
= m_splitter
->widget(0)->width() - leadingSpacing
- trailingSpacing
;
161 const int widthNeededForUrlNavigator
= primaryUrlNavigator()->sizeHint().width() - widthLeftForUrlNavigator
;
162 if (widthNeededForUrlNavigator
> 0) {
163 trailingSpacing
-= widthNeededForUrlNavigator
;
164 if (trailingSpacing
< 0) {
165 leadingSpacing
+= trailingSpacing
;
168 if (leadingSpacing
< 0) {
172 spacing(Primary
, Leading
)->setMinimumWidth(leadingSpacing
);
173 spacing(Primary
, Trailing
)->setFixedWidth(trailingSpacing
);
175 // secondary side of m_splitter
176 if (m_globalXOfSecondary
== INT_MIN
) {
177 Q_ASSERT(m_widthOfSecondary
== INT_MIN
);
180 spacing(Primary
, Trailing
)->setFixedWidth(0);
182 trailingSpacing
= (m_globalXOfSplitter
+ m_splitter
->width())
183 - (m_globalXOfSecondary
+ m_widthOfSecondary
);
184 if (trailingSpacing
< 0 || emptyTrashButton(Secondary
)->isVisible()) {
187 const int widthLeftForUrlNavigator2
= m_splitter
->widget(1)->width() - trailingSpacing
;
188 const int widthNeededForUrlNavigator2
= secondaryUrlNavigator()->sizeHint().width() - widthLeftForUrlNavigator2
;
189 if (widthNeededForUrlNavigator2
> 0) {
190 trailingSpacing
-= widthNeededForUrlNavigator2
;
191 if (trailingSpacing
< 0) {
196 spacing(Secondary
, Trailing
)->setMinimumWidth(trailingSpacing
);
199 QWidget
*DolphinNavigatorsWidgetAction::createNavigatorWidget(Side side
) const
201 auto navigatorWidget
= new QWidget(m_splitter
.get());
202 auto layout
= new QHBoxLayout
{navigatorWidget
};
203 layout
->setSpacing(0);
204 layout
->setContentsMargins(0, 0, 0, 0);
205 if (side
== Primary
) {
206 auto leadingSpacing
= new QWidget
{navigatorWidget
};
207 layout
->addWidget(leadingSpacing
);
209 auto urlNavigator
= new DolphinUrlNavigator(navigatorWidget
);
210 layout
->addWidget(urlNavigator
);
212 auto emptyTrashButton
= newEmptyTrashButton(urlNavigator
, navigatorWidget
);
213 layout
->addWidget(emptyTrashButton
);
215 connect(urlNavigator
, &KUrlNavigator::urlChanged
, this, [this]() {
216 // We have to wait for DolphinUrlNavigator::sizeHint() to update which
217 // happens a little bit later than when urlChanged is emitted.
218 this->m_adjustSpacingTimer
->start();
221 auto trailingSpacing
= new QWidget
{navigatorWidget
};
222 layout
->addWidget(trailingSpacing
);
223 return navigatorWidget
;
226 QPushButton
* DolphinNavigatorsWidgetAction::emptyTrashButton(DolphinNavigatorsWidgetAction::Side side
)
228 int sideIndex
= (side
== Primary
? 0 : 1);
229 if (side
== Primary
) {
230 return static_cast<QPushButton
*>(m_splitter
->widget(sideIndex
)->layout()->itemAt(2)->widget());
232 return static_cast<QPushButton
*>(m_splitter
->widget(sideIndex
)->layout()->itemAt(1)->widget());
235 QPushButton
*DolphinNavigatorsWidgetAction::newEmptyTrashButton(const DolphinUrlNavigator
*urlNavigator
, QWidget
*parent
) const
237 auto emptyTrashButton
= new QPushButton(QIcon::fromTheme(QStringLiteral("user-trash")),
238 i18nc("@action:button", "Empty Trash"), parent
);
239 emptyTrashButton
->setFlat(true);
240 connect(emptyTrashButton
, &QPushButton::clicked
,
241 this, [parent
]() { Trash::empty(parent
); });
242 connect(&Trash::instance(), &Trash::emptinessChanged
,
243 emptyTrashButton
, &QPushButton::setDisabled
);
244 emptyTrashButton
->hide();
245 connect(urlNavigator
, &KUrlNavigator::urlChanged
, this, [emptyTrashButton
, urlNavigator
]() {
246 emptyTrashButton
->setVisible(urlNavigator
->locationUrl().scheme() == QLatin1String("trash"));
248 emptyTrashButton
->setDisabled(Trash::isEmpty());
249 return emptyTrashButton
;
252 QWidget
*DolphinNavigatorsWidgetAction::spacing(Side side
, Position position
) const
254 int sideIndex
= (side
== Primary
? 0 : 1);
255 if (position
== Leading
) {
256 Q_ASSERT(side
== Primary
); // The secondary side of the splitter has no leading spacing.
257 return m_splitter
->widget(sideIndex
)->layout()->itemAt(0)->widget();
259 if (side
== Primary
) {
260 return m_splitter
->widget(sideIndex
)->layout()->itemAt(3)->widget();
262 return m_splitter
->widget(sideIndex
)->layout()->itemAt(2)->widget();
265 void DolphinNavigatorsWidgetAction::updateText()
267 const int urlNavigatorsAmount
= m_splitter
->count() > 1 && m_splitter
->widget(1)->isVisible() ?
269 setText(i18ncp("@action:inmenu", "Url Navigator", "Url Navigators", urlNavigatorsAmount
));