1 /***************************************************************************
2 * Copyright (C) 2014 by Emmanuel Pescosta <emmanuelpescosta099@gmail.com> *
4 * This program is free software; you can redistribute it and/or modify *
5 * it under the terms of the GNU General Public License as published by *
6 * the Free Software Foundation; either version 2 of the License, or *
7 * (at your option) any later version. *
9 * This program is distributed in the hope that it will be useful, *
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
12 * GNU General Public License for more details. *
14 * You should have received a copy of the GNU General Public License *
15 * along with this program; if not, write to the *
16 * Free Software Foundation, Inc., *
17 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA *
18 ***************************************************************************/
20 #include "dolphintabpage.h"
22 #include "dolphin_generalsettings.h"
23 #include "dolphinviewcontainer.h"
26 #include <QVBoxLayout>
28 DolphinTabPage::DolphinTabPage(const QUrl
&primaryUrl
, const QUrl
&secondaryUrl
, QWidget
* parent
) :
30 m_primaryViewActive(true),
31 m_splitViewEnabled(false),
34 QVBoxLayout
* layout
= new QVBoxLayout(this);
35 layout
->setSpacing(0);
38 m_splitter
= new QSplitter(Qt::Horizontal
, this);
39 m_splitter
->setChildrenCollapsible(false);
40 layout
->addWidget(m_splitter
);
42 // Create a new primary view
43 m_primaryViewContainer
= createViewContainer(primaryUrl
);
44 connect(m_primaryViewContainer
->view(), &DolphinView::urlChanged
,
45 this, &DolphinTabPage::activeViewUrlChanged
);
46 connect(m_primaryViewContainer
->view(), &DolphinView::redirection
,
47 this, &DolphinTabPage::slotViewUrlRedirection
);
49 m_splitter
->addWidget(m_primaryViewContainer
);
50 m_primaryViewContainer
->show();
52 if (secondaryUrl
.isValid() || GeneralSettings::splitView()) {
53 // Provide a secondary view, if the given secondary url is valid or if the
54 // startup settings are set this way (use the url of the primary view).
55 m_splitViewEnabled
= true;
56 const QUrl
& url
= secondaryUrl
.isValid() ? secondaryUrl
: primaryUrl
;
57 m_secondaryViewContainer
= createViewContainer(url
);
58 m_splitter
->addWidget(m_secondaryViewContainer
);
59 m_secondaryViewContainer
->show();
62 m_primaryViewContainer
->setActive(true);
65 bool DolphinTabPage::primaryViewActive() const
67 return m_primaryViewActive
;
70 bool DolphinTabPage::splitViewEnabled() const
72 return m_splitViewEnabled
;
75 void DolphinTabPage::setSplitViewEnabled(bool enabled
, const QUrl
&secondaryUrl
)
77 if (m_splitViewEnabled
!= enabled
) {
78 m_splitViewEnabled
= enabled
;
81 const QUrl
& url
= (secondaryUrl
.isEmpty()) ? m_primaryViewContainer
->url() : secondaryUrl
;
82 m_secondaryViewContainer
= createViewContainer(url
);
84 const bool placesSelectorVisible
= m_primaryViewContainer
->urlNavigator()->isPlacesSelectorVisible();
85 m_secondaryViewContainer
->urlNavigator()->setPlacesSelectorVisible(placesSelectorVisible
);
87 m_splitter
->addWidget(m_secondaryViewContainer
);
88 m_secondaryViewContainer
->show();
89 m_secondaryViewContainer
->setActive(true);
91 // Close the view which is active.
92 DolphinViewContainer
* view
= activeViewContainer();
93 if (m_primaryViewActive
) {
94 // If the primary view is active, we have to swap the pointers
95 // because the secondary view will be the new primary view.
96 qSwap(m_primaryViewContainer
, m_secondaryViewContainer
);
97 m_primaryViewActive
= false;
99 m_primaryViewContainer
->setActive(true);
106 DolphinViewContainer
* DolphinTabPage::primaryViewContainer() const
108 return m_primaryViewContainer
;
111 DolphinViewContainer
* DolphinTabPage::secondaryViewContainer() const
113 return m_secondaryViewContainer
;
116 DolphinViewContainer
* DolphinTabPage::activeViewContainer() const
118 return m_primaryViewActive
? m_primaryViewContainer
:
119 m_secondaryViewContainer
;
122 KFileItemList
DolphinTabPage::selectedItems() const
124 KFileItemList items
= m_primaryViewContainer
->view()->selectedItems();
125 if (m_splitViewEnabled
) {
126 items
+= m_secondaryViewContainer
->view()->selectedItems();
131 int DolphinTabPage::selectedItemsCount() const
133 int selectedItemsCount
= m_primaryViewContainer
->view()->selectedItemsCount();
134 if (m_splitViewEnabled
) {
135 selectedItemsCount
+= m_secondaryViewContainer
->view()->selectedItemsCount();
137 return selectedItemsCount
;
140 void DolphinTabPage::markUrlsAsSelected(const QList
<QUrl
>& urls
)
142 m_primaryViewContainer
->view()->markUrlsAsSelected(urls
);
143 if (m_splitViewEnabled
) {
144 m_secondaryViewContainer
->view()->markUrlsAsSelected(urls
);
148 void DolphinTabPage::markUrlAsCurrent(const QUrl
& url
)
150 m_primaryViewContainer
->view()->markUrlAsCurrent(url
);
151 if (m_splitViewEnabled
) {
152 m_secondaryViewContainer
->view()->markUrlAsCurrent(url
);
156 void DolphinTabPage::setPlacesSelectorVisible(bool visible
)
158 m_primaryViewContainer
->urlNavigator()->setPlacesSelectorVisible(visible
);
159 if (m_splitViewEnabled
) {
160 m_secondaryViewContainer
->urlNavigator()->setPlacesSelectorVisible(visible
);
164 void DolphinTabPage::refreshViews()
166 m_primaryViewContainer
->readSettings();
167 if (m_splitViewEnabled
) {
168 m_secondaryViewContainer
->readSettings();
172 QByteArray
DolphinTabPage::saveState() const
175 QDataStream
stream(&state
, QIODevice::WriteOnly
);
177 stream
<< quint32(2); // Tab state version
179 stream
<< m_splitViewEnabled
;
181 stream
<< m_primaryViewContainer
->url();
182 stream
<< m_primaryViewContainer
->urlNavigator()->isUrlEditable();
183 m_primaryViewContainer
->view()->saveState(stream
);
185 if (m_splitViewEnabled
) {
186 stream
<< m_secondaryViewContainer
->url();
187 stream
<< m_secondaryViewContainer
->urlNavigator()->isUrlEditable();
188 m_secondaryViewContainer
->view()->saveState(stream
);
191 stream
<< m_primaryViewActive
;
192 stream
<< m_splitter
->saveState();
197 void DolphinTabPage::restoreState(const QByteArray
& state
)
199 if (state
.isEmpty()) {
203 QByteArray sd
= state
;
204 QDataStream
stream(&sd
, QIODevice::ReadOnly
);
206 // Read the version number of the tab state and check if the version is supported.
210 // The version of the tab state isn't supported, we can't restore it.
214 bool isSplitViewEnabled
= false;
215 stream
>> isSplitViewEnabled
;
216 setSplitViewEnabled(isSplitViewEnabled
);
219 stream
>> primaryUrl
;
220 m_primaryViewContainer
->setUrl(primaryUrl
);
221 bool primaryUrlEditable
;
222 stream
>> primaryUrlEditable
;
223 m_primaryViewContainer
->urlNavigator()->setUrlEditable(primaryUrlEditable
);
224 m_primaryViewContainer
->view()->restoreState(stream
);
226 if (isSplitViewEnabled
) {
228 stream
>> secondaryUrl
;
229 m_secondaryViewContainer
->setUrl(secondaryUrl
);
230 bool secondaryUrlEditable
;
231 stream
>> secondaryUrlEditable
;
232 m_secondaryViewContainer
->urlNavigator()->setUrlEditable(secondaryUrlEditable
);
233 m_secondaryViewContainer
->view()->restoreState(stream
);
236 stream
>> m_primaryViewActive
;
237 if (m_primaryViewActive
) {
238 m_primaryViewContainer
->setActive(true);
240 Q_ASSERT(m_splitViewEnabled
);
241 m_secondaryViewContainer
->setActive(true);
244 QByteArray splitterState
;
245 stream
>> splitterState
;
246 m_splitter
->restoreState(splitterState
);
249 void DolphinTabPage::restoreStateV1(const QByteArray
& state
)
251 if (state
.isEmpty()) {
255 QByteArray sd
= state
;
256 QDataStream
stream(&sd
, QIODevice::ReadOnly
);
258 bool isSplitViewEnabled
= false;
259 stream
>> isSplitViewEnabled
;
260 setSplitViewEnabled(isSplitViewEnabled
);
263 stream
>> primaryUrl
;
264 m_primaryViewContainer
->setUrl(primaryUrl
);
265 bool primaryUrlEditable
;
266 stream
>> primaryUrlEditable
;
267 m_primaryViewContainer
->urlNavigator()->setUrlEditable(primaryUrlEditable
);
269 if (isSplitViewEnabled
) {
271 stream
>> secondaryUrl
;
272 m_secondaryViewContainer
->setUrl(secondaryUrl
);
273 bool secondaryUrlEditable
;
274 stream
>> secondaryUrlEditable
;
275 m_secondaryViewContainer
->urlNavigator()->setUrlEditable(secondaryUrlEditable
);
278 stream
>> m_primaryViewActive
;
279 if (m_primaryViewActive
) {
280 m_primaryViewContainer
->setActive(true);
282 Q_ASSERT(m_splitViewEnabled
);
283 m_secondaryViewContainer
->setActive(true);
286 QByteArray splitterState
;
287 stream
>> splitterState
;
288 m_splitter
->restoreState(splitterState
);
291 void DolphinTabPage::setActive(bool active
)
296 // we should bypass changing active view in split mode
297 m_active
= !m_splitViewEnabled
;
299 // we want view to fire activated when goes from false to true
300 activeViewContainer()->setActive(active
);
303 void DolphinTabPage::slotViewActivated()
305 const DolphinView
* oldActiveView
= activeViewContainer()->view();
307 // Set the view, which was active before, to inactive
308 // and update the active view type, if tab is active
310 if (m_splitViewEnabled
) {
311 activeViewContainer()->setActive(false);
312 m_primaryViewActive
= !m_primaryViewActive
;
314 m_primaryViewActive
= true;
315 if (m_secondaryViewContainer
) {
316 m_secondaryViewContainer
->setActive(false);
321 const DolphinView
* newActiveView
= activeViewContainer()->view();
323 if (newActiveView
!= oldActiveView
) {
324 disconnect(oldActiveView
, &DolphinView::urlChanged
,
325 this, &DolphinTabPage::activeViewUrlChanged
);
326 disconnect(oldActiveView
, &DolphinView::redirection
,
327 this, &DolphinTabPage::slotViewUrlRedirection
);
328 connect(newActiveView
, &DolphinView::urlChanged
,
329 this, &DolphinTabPage::activeViewUrlChanged
);
330 connect(newActiveView
, &DolphinView::redirection
,
331 this, &DolphinTabPage::slotViewUrlRedirection
);
334 emit
activeViewUrlChanged(activeViewContainer()->url());
335 emit
activeViewChanged(activeViewContainer());
338 void DolphinTabPage::slotViewUrlRedirection(const QUrl
& oldUrl
, const QUrl
& newUrl
)
342 emit
activeViewUrlChanged(newUrl
);
345 void DolphinTabPage::switchActiveView()
347 if (!m_splitViewEnabled
) {
350 if (m_primaryViewActive
) {
351 m_secondaryViewContainer
->setActive(true);
353 m_primaryViewContainer
->setActive(true);
357 DolphinViewContainer
* DolphinTabPage::createViewContainer(const QUrl
& url
) const
359 DolphinViewContainer
* container
= new DolphinViewContainer(url
, m_splitter
);
360 container
->setActive(false);
362 const DolphinView
* view
= container
->view();
363 connect(view
, &DolphinView::activated
,
364 this, &DolphinTabPage::slotViewActivated
);
366 connect(view
, &DolphinView::toggleActiveViewRequested
,
367 this, &DolphinTabPage::switchActiveView
);