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
);
98 m_primaryViewContainer
->setActive(true);
105 DolphinViewContainer
* DolphinTabPage::primaryViewContainer() const
107 return m_primaryViewContainer
;
110 DolphinViewContainer
* DolphinTabPage::secondaryViewContainer() const
112 return m_secondaryViewContainer
;
115 DolphinViewContainer
* DolphinTabPage::activeViewContainer() const
117 return m_primaryViewActive
? m_primaryViewContainer
:
118 m_secondaryViewContainer
;
121 KFileItemList
DolphinTabPage::selectedItems() const
123 KFileItemList items
= m_primaryViewContainer
->view()->selectedItems();
124 if (m_splitViewEnabled
) {
125 items
+= m_secondaryViewContainer
->view()->selectedItems();
130 int DolphinTabPage::selectedItemsCount() const
132 int selectedItemsCount
= m_primaryViewContainer
->view()->selectedItemsCount();
133 if (m_splitViewEnabled
) {
134 selectedItemsCount
+= m_secondaryViewContainer
->view()->selectedItemsCount();
136 return selectedItemsCount
;
139 void DolphinTabPage::markUrlsAsSelected(const QList
<QUrl
>& urls
)
141 m_primaryViewContainer
->view()->markUrlsAsSelected(urls
);
142 if (m_splitViewEnabled
) {
143 m_secondaryViewContainer
->view()->markUrlsAsSelected(urls
);
147 void DolphinTabPage::markUrlAsCurrent(const QUrl
& url
)
149 m_primaryViewContainer
->view()->markUrlAsCurrent(url
);
150 if (m_splitViewEnabled
) {
151 m_secondaryViewContainer
->view()->markUrlAsCurrent(url
);
155 void DolphinTabPage::setPlacesSelectorVisible(bool visible
)
157 m_primaryViewContainer
->urlNavigator()->setPlacesSelectorVisible(visible
);
158 if (m_splitViewEnabled
) {
159 m_secondaryViewContainer
->urlNavigator()->setPlacesSelectorVisible(visible
);
163 void DolphinTabPage::refreshViews()
165 m_primaryViewContainer
->readSettings();
166 if (m_splitViewEnabled
) {
167 m_secondaryViewContainer
->readSettings();
171 QByteArray
DolphinTabPage::saveState() const
174 QDataStream
stream(&state
, QIODevice::WriteOnly
);
176 stream
<< quint32(2); // Tab state version
178 stream
<< m_splitViewEnabled
;
180 stream
<< m_primaryViewContainer
->url();
181 stream
<< m_primaryViewContainer
->urlNavigator()->isUrlEditable();
182 m_primaryViewContainer
->view()->saveState(stream
);
184 if (m_splitViewEnabled
) {
185 stream
<< m_secondaryViewContainer
->url();
186 stream
<< m_secondaryViewContainer
->urlNavigator()->isUrlEditable();
187 m_secondaryViewContainer
->view()->saveState(stream
);
190 stream
<< m_primaryViewActive
;
191 stream
<< m_splitter
->saveState();
196 void DolphinTabPage::restoreState(const QByteArray
& state
)
198 if (state
.isEmpty()) {
202 QByteArray sd
= state
;
203 QDataStream
stream(&sd
, QIODevice::ReadOnly
);
205 // Read the version number of the tab state and check if the version is supported.
209 // The version of the tab state isn't supported, we can't restore it.
213 bool isSplitViewEnabled
= false;
214 stream
>> isSplitViewEnabled
;
215 setSplitViewEnabled(isSplitViewEnabled
);
218 stream
>> primaryUrl
;
219 m_primaryViewContainer
->setUrl(primaryUrl
);
220 bool primaryUrlEditable
;
221 stream
>> primaryUrlEditable
;
222 m_primaryViewContainer
->urlNavigator()->setUrlEditable(primaryUrlEditable
);
223 m_primaryViewContainer
->view()->restoreState(stream
);
225 if (isSplitViewEnabled
) {
227 stream
>> secondaryUrl
;
228 m_secondaryViewContainer
->setUrl(secondaryUrl
);
229 bool secondaryUrlEditable
;
230 stream
>> secondaryUrlEditable
;
231 m_secondaryViewContainer
->urlNavigator()->setUrlEditable(secondaryUrlEditable
);
232 m_secondaryViewContainer
->view()->restoreState(stream
);
235 stream
>> m_primaryViewActive
;
236 if (m_primaryViewActive
) {
237 m_primaryViewContainer
->setActive(true);
239 Q_ASSERT(m_splitViewEnabled
);
240 m_secondaryViewContainer
->setActive(true);
243 QByteArray splitterState
;
244 stream
>> splitterState
;
245 m_splitter
->restoreState(splitterState
);
248 void DolphinTabPage::restoreStateV1(const QByteArray
& state
)
250 if (state
.isEmpty()) {
254 QByteArray sd
= state
;
255 QDataStream
stream(&sd
, QIODevice::ReadOnly
);
257 bool isSplitViewEnabled
= false;
258 stream
>> isSplitViewEnabled
;
259 setSplitViewEnabled(isSplitViewEnabled
);
262 stream
>> primaryUrl
;
263 m_primaryViewContainer
->setUrl(primaryUrl
);
264 bool primaryUrlEditable
;
265 stream
>> primaryUrlEditable
;
266 m_primaryViewContainer
->urlNavigator()->setUrlEditable(primaryUrlEditable
);
268 if (isSplitViewEnabled
) {
270 stream
>> secondaryUrl
;
271 m_secondaryViewContainer
->setUrl(secondaryUrl
);
272 bool secondaryUrlEditable
;
273 stream
>> secondaryUrlEditable
;
274 m_secondaryViewContainer
->urlNavigator()->setUrlEditable(secondaryUrlEditable
);
277 stream
>> m_primaryViewActive
;
278 if (m_primaryViewActive
) {
279 m_primaryViewContainer
->setActive(true);
281 Q_ASSERT(m_splitViewEnabled
);
282 m_secondaryViewContainer
->setActive(true);
285 QByteArray splitterState
;
286 stream
>> splitterState
;
287 m_splitter
->restoreState(splitterState
);
290 void DolphinTabPage::setActive(bool active
)
295 // we should bypass changing active view in split mode
296 m_active
= !m_splitViewEnabled
;
298 // we want view to fire activated when goes from false to true
299 activeViewContainer()->setActive(active
);
302 void DolphinTabPage::slotViewActivated()
304 const DolphinView
* oldActiveView
= activeViewContainer()->view();
306 // Set the view, which was active before, to inactive
307 // and update the active view type, if tab is active
309 if (m_splitViewEnabled
) {
310 activeViewContainer()->setActive(false);
311 m_primaryViewActive
= !m_primaryViewActive
;
313 m_primaryViewActive
= true;
317 const DolphinView
* newActiveView
= activeViewContainer()->view();
319 if (newActiveView
!= oldActiveView
) {
320 disconnect(oldActiveView
, &DolphinView::urlChanged
,
321 this, &DolphinTabPage::activeViewUrlChanged
);
322 disconnect(oldActiveView
, &DolphinView::redirection
,
323 this, &DolphinTabPage::slotViewUrlRedirection
);
324 connect(newActiveView
, &DolphinView::urlChanged
,
325 this, &DolphinTabPage::activeViewUrlChanged
);
326 connect(newActiveView
, &DolphinView::redirection
,
327 this, &DolphinTabPage::slotViewUrlRedirection
);
330 emit
activeViewUrlChanged(activeViewContainer()->url());
331 emit
activeViewChanged(activeViewContainer());
334 void DolphinTabPage::slotViewUrlRedirection(const QUrl
& oldUrl
, const QUrl
& newUrl
)
338 emit
activeViewUrlChanged(newUrl
);
341 void DolphinTabPage::switchActiveView()
343 if (!m_splitViewEnabled
) {
346 if (m_primaryViewActive
) {
347 m_secondaryViewContainer
->setActive(true);
349 m_primaryViewContainer
->setActive(true);
353 DolphinViewContainer
* DolphinTabPage::createViewContainer(const QUrl
& url
) const
355 DolphinViewContainer
* container
= new DolphinViewContainer(url
, m_splitter
);
356 container
->setActive(false);
358 const DolphinView
* view
= container
->view();
359 connect(view
, &DolphinView::activated
,
360 this, &DolphinTabPage::slotViewActivated
);
362 connect(view
, &DolphinView::toggleActiveViewRequested
,
363 this, &DolphinTabPage::switchActiveView
);