]> cloud.milkyroute.net Git - dolphin.git/blob - src/dolphintabpage.cpp
f6d288e7d1b3de4ba161c6fe170cf92bcb2ddc1e
[dolphin.git] / src / dolphintabpage.cpp
1 /*
2 * SPDX-FileCopyrightText: 2014 Emmanuel Pescosta <emmanuelpescosta099@gmail.com>
3 * SPDX-FileCopyrightText: 2020 Felix Ernst <fe.a.ernst@gmail.com>
4 *
5 * SPDX-License-Identifier: GPL-2.0-or-later
6 */
7
8 #include "dolphintabpage.h"
9
10 #include "dolphin_generalsettings.h"
11 #include "dolphinviewcontainer.h"
12
13 #include <QGridLayout>
14 #include <QStyle>
15 #include <QVariantAnimation>
16
17 DolphinTabPage::DolphinTabPage(const QUrl &primaryUrl, const QUrl &secondaryUrl, QWidget* parent) :
18 QWidget(parent),
19 m_expandingContainer{nullptr},
20 m_primaryViewActive(true),
21 m_splitViewEnabled(false),
22 m_active(true)
23 {
24 QGridLayout *layout = new QGridLayout(this);
25 layout->setSpacing(0);
26 layout->setContentsMargins(0, 0, 0, 0);
27
28 m_splitter = new DolphinTabPageSplitter(Qt::Horizontal, this);
29 m_splitter->setChildrenCollapsible(false);
30 connect(m_splitter, &QSplitter::splitterMoved,
31 this, &DolphinTabPage::splitterMoved);
32 layout->addWidget(m_splitter, 1, 0);
33 layout->setRowStretch(1, 1);
34
35 // Create a new primary view
36 m_primaryViewContainer = createViewContainer(primaryUrl);
37 connect(m_primaryViewContainer->view(), &DolphinView::urlChanged,
38 this, &DolphinTabPage::activeViewUrlChanged);
39 connect(m_primaryViewContainer->view(), &DolphinView::redirection,
40 this, &DolphinTabPage::slotViewUrlRedirection);
41
42 m_splitter->addWidget(m_primaryViewContainer);
43 m_primaryViewContainer->show();
44
45 if (secondaryUrl.isValid() || GeneralSettings::splitView()) {
46 // Provide a secondary view, if the given secondary url is valid or if the
47 // startup settings are set this way (use the url of the primary view).
48 m_splitViewEnabled = true;
49 const QUrl& url = secondaryUrl.isValid() ? secondaryUrl : primaryUrl;
50 m_secondaryViewContainer = createViewContainer(url);
51 m_splitter->addWidget(m_secondaryViewContainer);
52 m_secondaryViewContainer->show();
53 }
54
55 m_primaryViewContainer->setActive(true);
56 }
57
58 bool DolphinTabPage::primaryViewActive() const
59 {
60 return m_primaryViewActive;
61 }
62
63 bool DolphinTabPage::splitViewEnabled() const
64 {
65 return m_splitViewEnabled;
66 }
67
68 void DolphinTabPage::setSplitViewEnabled(bool enabled, Animated animated, const QUrl &secondaryUrl)
69 {
70 if (m_splitViewEnabled != enabled) {
71 m_splitViewEnabled = enabled;
72 if (animated == WithAnimation && (
73 style()->styleHint(QStyle::SH_Widget_Animation_Duration, nullptr, this) < 1 ||
74 GlobalConfig::animationDurationFactor() <= 0.0)) {
75 animated = WithoutAnimation;
76 }
77 if (m_expandViewAnimation) {
78 m_expandViewAnimation->stop(); // deletes because of QAbstractAnimation::DeleteWhenStopped.
79 if (animated == WithoutAnimation) {
80 slotAnimationFinished();
81 }
82 }
83
84 if (enabled) {
85 QList<int> splitterSizes = m_splitter->sizes();
86 const QUrl& url = (secondaryUrl.isEmpty()) ? m_primaryViewContainer->url() : secondaryUrl;
87 m_secondaryViewContainer = createViewContainer(url);
88
89 auto secondaryNavigator = m_navigatorsWidget->secondaryUrlNavigator();
90 if (!secondaryNavigator) {
91 m_navigatorsWidget->createSecondaryUrlNavigator();
92 secondaryNavigator = m_navigatorsWidget->secondaryUrlNavigator();
93 }
94 m_secondaryViewContainer->connectUrlNavigator(secondaryNavigator);
95 m_navigatorsWidget->setSecondaryNavigatorVisible(true);
96 m_navigatorsWidget->followViewContainersGeometry(m_primaryViewContainer,
97 m_secondaryViewContainer);
98
99 m_splitter->addWidget(m_secondaryViewContainer);
100 m_secondaryViewContainer->setActive(true);
101
102 if (animated == WithAnimation) {
103 m_secondaryViewContainer->setMinimumWidth(1);
104 splitterSizes.append(1);
105 m_splitter->setSizes(splitterSizes);
106 startExpandViewAnimation(m_secondaryViewContainer);
107 }
108 m_secondaryViewContainer->show();
109 } else {
110 m_navigatorsWidget->setSecondaryNavigatorVisible(false);
111 m_secondaryViewContainer->disconnectUrlNavigator();
112
113 DolphinViewContainer* view;
114 if (GeneralSettings::closeActiveSplitView()) {
115 view = activeViewContainer();
116 if (m_primaryViewActive) {
117 m_primaryViewContainer->disconnectUrlNavigator();
118 m_secondaryViewContainer->connectUrlNavigator(
119 m_navigatorsWidget->primaryUrlNavigator());
120
121 // If the primary view is active, we have to swap the pointers
122 // because the secondary view will be the new primary view.
123 std::swap(m_primaryViewContainer, m_secondaryViewContainer);
124 m_primaryViewActive = false;
125 }
126 } else {
127 view = m_primaryViewActive ? m_secondaryViewContainer : m_primaryViewContainer;
128 if (!m_primaryViewActive) {
129 m_primaryViewContainer->disconnectUrlNavigator();
130 m_secondaryViewContainer->connectUrlNavigator(
131 m_navigatorsWidget->primaryUrlNavigator());
132
133 // If the secondary view is active, we have to swap the pointers
134 // because the secondary view will be the new primary view.
135 std::swap(m_primaryViewContainer, m_secondaryViewContainer);
136 m_primaryViewActive = true;
137 }
138 }
139 m_primaryViewContainer->setActive(true);
140 m_navigatorsWidget->followViewContainersGeometry(m_primaryViewContainer, nullptr);
141
142 if (animated == WithoutAnimation) {
143 view->close();
144 view->deleteLater();
145 } else {
146 // Kill it but keep it as a zombie for the closing animation.
147 m_secondaryViewContainer = nullptr;
148 view->blockSignals(true);
149 view->view()->blockSignals(true);
150 view->setDisabled(true);
151 startExpandViewAnimation(m_primaryViewContainer);
152 }
153
154 m_primaryViewContainer->slotSplitTabDisabled();
155 }
156 }
157 }
158
159 DolphinViewContainer* DolphinTabPage::primaryViewContainer() const
160 {
161 return m_primaryViewContainer;
162 }
163
164 DolphinViewContainer* DolphinTabPage::secondaryViewContainer() const
165 {
166 return m_secondaryViewContainer;
167 }
168
169 DolphinViewContainer* DolphinTabPage::activeViewContainer() const
170 {
171 return m_primaryViewActive ? m_primaryViewContainer :
172 m_secondaryViewContainer;
173 }
174
175 KFileItemList DolphinTabPage::selectedItems() const
176 {
177 KFileItemList items = m_primaryViewContainer->view()->selectedItems();
178 if (m_splitViewEnabled) {
179 items += m_secondaryViewContainer->view()->selectedItems();
180 }
181 return items;
182 }
183
184 int DolphinTabPage::selectedItemsCount() const
185 {
186 int selectedItemsCount = m_primaryViewContainer->view()->selectedItemsCount();
187 if (m_splitViewEnabled) {
188 selectedItemsCount += m_secondaryViewContainer->view()->selectedItemsCount();
189 }
190 return selectedItemsCount;
191 }
192
193 void DolphinTabPage::connectNavigators(DolphinNavigatorsWidgetAction *navigatorsWidget)
194 {
195 insertNavigatorsWidget(navigatorsWidget);
196 m_navigatorsWidget = navigatorsWidget;
197 auto primaryNavigator = navigatorsWidget->primaryUrlNavigator();
198 m_primaryViewContainer->connectUrlNavigator(primaryNavigator);
199 if (m_splitViewEnabled) {
200 auto secondaryNavigator = navigatorsWidget->secondaryUrlNavigator();
201 m_secondaryViewContainer->connectUrlNavigator(secondaryNavigator);
202 }
203 m_navigatorsWidget->followViewContainersGeometry(m_primaryViewContainer,
204 m_secondaryViewContainer);
205 }
206
207 void DolphinTabPage::disconnectNavigators()
208 {
209 m_navigatorsWidget = nullptr;
210 m_primaryViewContainer->disconnectUrlNavigator();
211 if (m_splitViewEnabled) {
212 m_secondaryViewContainer->disconnectUrlNavigator();
213 }
214 }
215
216 void DolphinTabPage::insertNavigatorsWidget(DolphinNavigatorsWidgetAction* navigatorsWidget)
217 {
218 QGridLayout *gridLayout = static_cast<QGridLayout *>(layout());
219 if (navigatorsWidget->isInToolbar()) {
220 gridLayout->setRowMinimumHeight(0, 0);
221 } else {
222 // We set a row minimum height, so the height does not visibly change whenever
223 // navigatorsWidget is inserted which happens every time the current tab is changed.
224 gridLayout->setRowMinimumHeight(0, navigatorsWidget->primaryUrlNavigator()->height());
225 gridLayout->addWidget(navigatorsWidget->requestWidget(this), 0, 0);
226 }
227 }
228
229 void DolphinTabPage::markUrlsAsSelected(const QList<QUrl>& urls)
230 {
231 m_primaryViewContainer->view()->markUrlsAsSelected(urls);
232 if (m_splitViewEnabled) {
233 m_secondaryViewContainer->view()->markUrlsAsSelected(urls);
234 }
235 }
236
237 void DolphinTabPage::markUrlAsCurrent(const QUrl& url)
238 {
239 m_primaryViewContainer->view()->markUrlAsCurrent(url);
240 if (m_splitViewEnabled) {
241 m_secondaryViewContainer->view()->markUrlAsCurrent(url);
242 }
243 }
244
245 void DolphinTabPage::refreshViews()
246 {
247 m_primaryViewContainer->readSettings();
248 if (m_splitViewEnabled) {
249 m_secondaryViewContainer->readSettings();
250 }
251 }
252
253 QByteArray DolphinTabPage::saveState() const
254 {
255 QByteArray state;
256 QDataStream stream(&state, QIODevice::WriteOnly);
257
258 stream << quint32(2); // Tab state version
259
260 stream << m_splitViewEnabled;
261
262 stream << m_primaryViewContainer->url();
263 stream << m_primaryViewContainer->urlNavigatorInternalWithHistory()->isUrlEditable();
264 m_primaryViewContainer->view()->saveState(stream);
265
266 if (m_splitViewEnabled) {
267 stream << m_secondaryViewContainer->url();
268 stream << m_secondaryViewContainer->urlNavigatorInternalWithHistory()->isUrlEditable();
269 m_secondaryViewContainer->view()->saveState(stream);
270 }
271
272 stream << m_primaryViewActive;
273 stream << m_splitter->saveState();
274
275 return state;
276 }
277
278 void DolphinTabPage::restoreState(const QByteArray& state)
279 {
280 if (state.isEmpty()) {
281 return;
282 }
283
284 QByteArray sd = state;
285 QDataStream stream(&sd, QIODevice::ReadOnly);
286
287 // Read the version number of the tab state and check if the version is supported.
288 quint32 version = 0;
289 stream >> version;
290 if (version != 2) {
291 // The version of the tab state isn't supported, we can't restore it.
292 return;
293 }
294
295 bool isSplitViewEnabled = false;
296 stream >> isSplitViewEnabled;
297 setSplitViewEnabled(isSplitViewEnabled, WithoutAnimation);
298
299 QUrl primaryUrl;
300 stream >> primaryUrl;
301 m_primaryViewContainer->setUrl(primaryUrl);
302 bool primaryUrlEditable;
303 stream >> primaryUrlEditable;
304 m_primaryViewContainer->urlNavigatorInternalWithHistory()->setUrlEditable(primaryUrlEditable);
305 m_primaryViewContainer->view()->restoreState(stream);
306
307 if (isSplitViewEnabled) {
308 QUrl secondaryUrl;
309 stream >> secondaryUrl;
310 m_secondaryViewContainer->setUrl(secondaryUrl);
311 bool secondaryUrlEditable;
312 stream >> secondaryUrlEditable;
313 m_secondaryViewContainer->urlNavigatorInternalWithHistory()->setUrlEditable(secondaryUrlEditable);
314 m_secondaryViewContainer->view()->restoreState(stream);
315 }
316
317 stream >> m_primaryViewActive;
318 if (m_primaryViewActive) {
319 m_primaryViewContainer->setActive(true);
320 } else {
321 Q_ASSERT(m_splitViewEnabled);
322 m_secondaryViewContainer->setActive(true);
323 }
324
325 QByteArray splitterState;
326 stream >> splitterState;
327 m_splitter->restoreState(splitterState);
328 }
329
330 void DolphinTabPage::setActive(bool active)
331 {
332 if (active) {
333 m_active = active;
334 } else {
335 // we should bypass changing active view in split mode
336 m_active = !m_splitViewEnabled;
337 }
338 // we want view to fire activated when goes from false to true
339 activeViewContainer()->setActive(active);
340 }
341
342 void DolphinTabPage::slotAnimationFinished()
343 {
344 for (int i = 0; i < m_splitter->count(); ++i) {
345 QWidget *viewContainer = m_splitter->widget(i);
346 if (viewContainer != m_primaryViewContainer &&
347 viewContainer != m_secondaryViewContainer) {
348 viewContainer->close();
349 viewContainer->deleteLater();
350 }
351 }
352 for (int i = 0; i < m_splitter->count(); ++i) {
353 QWidget *viewContainer = m_splitter->widget(i);
354 viewContainer->setMinimumWidth(viewContainer->minimumSizeHint().width());
355 }
356 m_expandingContainer = nullptr;
357 }
358
359 void DolphinTabPage::slotAnimationValueChanged(const QVariant& value)
360 {
361 Q_CHECK_PTR(m_expandingContainer);
362 const int indexOfExpandingContainer = m_splitter->indexOf(m_expandingContainer);
363 int indexOfNonExpandingContainer = -1;
364 if (m_expandingContainer == m_primaryViewContainer) {
365 indexOfNonExpandingContainer = m_splitter->indexOf(m_secondaryViewContainer);
366 } else {
367 indexOfNonExpandingContainer = m_splitter->indexOf(m_primaryViewContainer);
368 }
369 std::vector<QWidget *> widgetsToRemove;
370 const QList<int> oldSplitterSizes = m_splitter->sizes();
371 QList<int> newSplitterSizes{oldSplitterSizes};
372 int expansionWidthNeeded = value.toInt() - oldSplitterSizes.at(indexOfExpandingContainer);
373
374 // Reduce the size of the other widgets to make space for the expandingContainer.
375 for (int i = m_splitter->count() - 1; i >= 0; --i) {
376 if (m_splitter->widget(i) == m_primaryViewContainer ||
377 m_splitter->widget(i) == m_secondaryViewContainer) {
378 continue;
379 }
380 newSplitterSizes[i] = oldSplitterSizes.at(i) - expansionWidthNeeded;
381 expansionWidthNeeded = 0;
382 if (indexOfNonExpandingContainer != -1) {
383 // Make sure every zombie container is at least slightly reduced in size
384 // so it doesn't seem like they are here to stay.
385 newSplitterSizes[i]--;
386 newSplitterSizes[indexOfNonExpandingContainer]++;
387 }
388 if (newSplitterSizes.at(i) <= 0) {
389 expansionWidthNeeded -= newSplitterSizes.at(i);
390 newSplitterSizes[i] = 0;
391 widgetsToRemove.emplace_back(m_splitter->widget(i));
392 }
393 }
394 if (expansionWidthNeeded > 1 && indexOfNonExpandingContainer != -1) {
395 Q_ASSERT(m_splitViewEnabled);
396 newSplitterSizes[indexOfNonExpandingContainer] -= expansionWidthNeeded;
397 }
398 newSplitterSizes[indexOfExpandingContainer] = value.toInt();
399 m_splitter->setSizes(newSplitterSizes);
400 while (!widgetsToRemove.empty()) {
401 widgetsToRemove.back()->close();
402 widgetsToRemove.back()->deleteLater();
403 widgetsToRemove.pop_back();
404 }
405 }
406
407
408 void DolphinTabPage::slotViewActivated()
409 {
410 const DolphinView* oldActiveView = activeViewContainer()->view();
411
412 // Set the view, which was active before, to inactive
413 // and update the active view type, if tab is active
414 if (m_active) {
415 if (m_splitViewEnabled) {
416 activeViewContainer()->setActive(false);
417 m_primaryViewActive = !m_primaryViewActive;
418 } else {
419 m_primaryViewActive = true;
420 if (m_secondaryViewContainer) {
421 m_secondaryViewContainer->setActive(false);
422 }
423 }
424 }
425
426 const DolphinView* newActiveView = activeViewContainer()->view();
427
428 if (newActiveView == oldActiveView) {
429 return;
430 }
431
432 disconnect(oldActiveView, &DolphinView::urlChanged,
433 this, &DolphinTabPage::activeViewUrlChanged);
434 disconnect(oldActiveView, &DolphinView::redirection,
435 this, &DolphinTabPage::slotViewUrlRedirection);
436 connect(newActiveView, &DolphinView::urlChanged,
437 this, &DolphinTabPage::activeViewUrlChanged);
438 connect(newActiveView, &DolphinView::redirection,
439 this, &DolphinTabPage::slotViewUrlRedirection);
440 Q_EMIT activeViewChanged(activeViewContainer());
441 Q_EMIT activeViewUrlChanged(activeViewContainer()->url());
442 }
443
444 void DolphinTabPage::slotViewUrlRedirection(const QUrl& oldUrl, const QUrl& newUrl)
445 {
446 Q_UNUSED(oldUrl)
447
448 Q_EMIT activeViewUrlChanged(newUrl);
449 }
450
451 void DolphinTabPage::switchActiveView()
452 {
453 if (!m_splitViewEnabled) {
454 return;
455 }
456 if (m_primaryViewActive) {
457 m_secondaryViewContainer->setActive(true);
458 } else {
459 m_primaryViewContainer->setActive(true);
460 }
461 }
462
463 DolphinViewContainer* DolphinTabPage::createViewContainer(const QUrl& url) const
464 {
465 DolphinViewContainer* container = new DolphinViewContainer(url, m_splitter);
466 container->setActive(false);
467
468 const DolphinView* view = container->view();
469 connect(view, &DolphinView::activated,
470 this, &DolphinTabPage::slotViewActivated);
471
472 connect(view, &DolphinView::toggleActiveViewRequested,
473 this, &DolphinTabPage::switchActiveView);
474
475 return container;
476 }
477
478 void DolphinTabPage::startExpandViewAnimation(DolphinViewContainer *expandingContainer)
479 {
480 Q_CHECK_PTR(expandingContainer);
481 Q_ASSERT(expandingContainer == m_primaryViewContainer ||
482 expandingContainer == m_secondaryViewContainer);
483 m_expandingContainer = expandingContainer;
484
485 m_expandViewAnimation = new QVariantAnimation(m_splitter);
486 m_expandViewAnimation->setDuration(2 *
487 style()->styleHint(QStyle::SH_Widget_Animation_Duration, nullptr, this) *
488 GlobalConfig::animationDurationFactor());
489 for (int i = 0; i < m_splitter->count(); ++i) {
490 m_splitter->widget(i)->setMinimumWidth(1);
491 }
492 connect(m_expandViewAnimation, &QAbstractAnimation::finished,
493 this, &DolphinTabPage::slotAnimationFinished);
494 connect(m_expandViewAnimation, &QVariantAnimation::valueChanged,
495 this, &DolphinTabPage::slotAnimationValueChanged);
496
497 m_expandViewAnimation->setStartValue(expandingContainer->width());
498 if (m_splitViewEnabled) { // A new viewContainer is being opened.
499 m_expandViewAnimation->setEndValue(m_splitter->width() / 2);
500 m_expandViewAnimation->setEasingCurve(QEasingCurve::OutCubic);
501 } else { // A viewContainer is being closed.
502 m_expandViewAnimation->setEndValue(m_splitter->width());
503 m_expandViewAnimation->setEasingCurve(QEasingCurve::InCubic);
504 }
505 m_expandViewAnimation->start(QAbstractAnimation::DeleteWhenStopped);
506 }
507
508 DolphinTabPageSplitterHandle::DolphinTabPageSplitterHandle(Qt::Orientation orientation, QSplitter *parent)
509 : QSplitterHandle(orientation, parent)
510 , m_mouseReleaseWasReceived(false)
511 {}
512
513 bool DolphinTabPageSplitterHandle::event(QEvent *event)
514 {
515 switch (event->type()) {
516 case QEvent::MouseButtonPress:
517 m_mouseReleaseWasReceived = false;
518 break;
519 case QEvent::MouseButtonRelease:
520 if (m_mouseReleaseWasReceived) {
521 resetSplitterSizes();
522 }
523 m_mouseReleaseWasReceived = !m_mouseReleaseWasReceived;
524 break;
525 case QEvent::MouseButtonDblClick:
526 m_mouseReleaseWasReceived = false;
527 resetSplitterSizes();
528 break;
529 default:
530 break;
531 }
532
533 return QSplitterHandle::event(event);
534 }
535
536 void DolphinTabPageSplitterHandle::resetSplitterSizes()
537 {
538 QList<int> splitterSizes = splitter()->sizes();
539 std::fill(splitterSizes.begin(), splitterSizes.end(), 0);
540 splitter()->setSizes(splitterSizes);
541 }
542
543 DolphinTabPageSplitter::DolphinTabPageSplitter(Qt::Orientation orientation, QWidget *parent)
544 : QSplitter(orientation, parent)
545 {}
546
547 QSplitterHandle* DolphinTabPageSplitter::createHandle()
548 {
549 return new DolphinTabPageSplitterHandle(orientation(), this);
550 }