source: calamares/trunk/fuentes/src/modules/partition/gui/PartitionViewStep.cpp @ 7538

Last change on this file since 7538 was 7538, checked in by kbut, 13 months ago

sync with github

File size: 20.8 KB
Line 
1/* === This file is part of Calamares - <https://github.com/calamares> ===
2 *
3 *   Copyright 2014, Aurélien Gâteau <agateau@kde.org>
4 *   Copyright 2014-2017, Teo Mrnjavac <teo@kde.org>
5 *
6 *   Calamares is free software: you can redistribute it and/or modify
7 *   it under the terms of the GNU General Public License as published by
8 *   the Free Software Foundation, either version 3 of the License, or
9 *   (at your option) any later version.
10 *
11 *   Calamares is distributed in the hope that it will be useful,
12 *   but WITHOUT ANY WARRANTY; without even the implied warranty of
13 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 *   GNU General Public License for more details.
15 *
16 *   You should have received a copy of the GNU General Public License
17 *   along with Calamares. If not, see <http://www.gnu.org/licenses/>.
18 */
19
20#include "gui/PartitionViewStep.h"
21
22#include "core/DeviceModel.h"
23#include "core/PartitionCoreModule.h"
24#include "core/PartitionModel.h"
25#include "core/KPMHelpers.h"
26#include "core/OsproberEntry.h"
27#include "core/PartUtils.h"
28#include "gui/ChoicePage.h"
29#include "gui/PartitionPage.h"
30#include "gui/PartitionBarsView.h"
31#include "gui/PartitionLabelsView.h"
32
33#include "CalamaresVersion.h"
34#include "utils/CalamaresUtilsGui.h"
35#include "utils/Logger.h"
36#include "utils/Retranslator.h"
37#include "widgets/WaitingWidget.h"
38#include "GlobalStorage.h"
39#include "JobQueue.h"
40#include "Job.h"
41#include "Branding.h"
42
43#include <kpmcore/core/device.h>
44#include <kpmcore/core/partition.h>
45#include <kpmcore/fs/filesystem.h>
46
47// Qt
48#include <QApplication>
49#include <QDir>
50#include <QFormLayout>
51#include <QLabel>
52#include <QMessageBox>
53#include <QProcess>
54#include <QStackedWidget>
55#include <QTimer>
56#include <QtConcurrent/QtConcurrent>
57#include <QFutureWatcher>
58
59PartitionViewStep::PartitionViewStep( QObject* parent )
60    : Calamares::ViewStep( parent )
61    , m_core( nullptr )
62    , m_widget( new QStackedWidget() )
63    , m_choicePage( nullptr )
64    , m_manualPartitionPage( nullptr )
65{
66    m_widget->setContentsMargins( 0, 0, 0, 0 );
67
68    m_waitingWidget = new WaitingWidget( QString() );
69    m_widget->addWidget( m_waitingWidget );
70    CALAMARES_RETRANSLATE( qobject_cast< WaitingWidget* >( m_waitingWidget )->setText( tr( "Gathering system information..." ) ); )
71
72    m_core = new PartitionCoreModule( this ); // Unusable before init is complete!
73    // We're not done loading, but we need the configuration map first.
74}
75
76
77void
78PartitionViewStep::initPartitionCoreModule()
79{
80    Q_ASSERT( m_core );
81    m_core->init();
82}
83
84
85void
86PartitionViewStep::continueLoading()
87{
88    Q_ASSERT( !m_choicePage );
89    Q_ASSERT( !m_manualPartitionPage );
90
91    m_manualPartitionPage = new PartitionPage( m_core );
92    m_choicePage = new ChoicePage();
93
94    m_choicePage->init( m_core );
95
96    m_widget->addWidget( m_choicePage );
97    m_widget->addWidget( m_manualPartitionPage );
98    m_widget->removeWidget( m_waitingWidget );
99    m_waitingWidget->deleteLater();
100    m_waitingWidget = nullptr;
101
102    connect( m_core,            &PartitionCoreModule::hasRootMountPointChanged,
103             this,              &PartitionViewStep::nextStatusChanged );
104    connect( m_choicePage,      &ChoicePage::nextStatusChanged,
105             this,              &PartitionViewStep::nextStatusChanged );
106}
107
108
109PartitionViewStep::~PartitionViewStep()
110{
111    if ( m_choicePage && m_choicePage->parent() == nullptr )
112        m_choicePage->deleteLater();
113    if ( m_manualPartitionPage && m_manualPartitionPage->parent() == nullptr )
114        m_manualPartitionPage->deleteLater();
115}
116
117
118QString
119PartitionViewStep::prettyName() const
120{
121    return tr( "Partitions" );
122}
123
124
125QWidget*
126PartitionViewStep::widget()
127{
128    return m_widget;
129}
130
131
132QWidget*
133PartitionViewStep::createSummaryWidget() const
134{
135    QWidget* widget = new QWidget;
136    QVBoxLayout* mainLayout = new QVBoxLayout;
137    widget->setLayout( mainLayout );
138    mainLayout->setMargin( 0 );
139
140    ChoicePage::Choice choice = m_choicePage->currentChoice();
141
142    QFormLayout* formLayout = new QFormLayout( widget );
143    const int MARGIN = CalamaresUtils::defaultFontHeight() / 2;
144    formLayout->setContentsMargins( MARGIN, 0, MARGIN, MARGIN );
145    mainLayout->addLayout( formLayout );
146
147    QList< PartitionCoreModule::SummaryInfo > list = m_core->createSummaryInfo();
148    if ( list.length() > 1 ) // There are changes on more than one disk
149    {
150        //NOTE: all of this should only happen when Manual partitioning is active.
151        //      Any other choice should result in a list.length() == 1.
152        QLabel* modeLabel = new QLabel;
153        formLayout->addRow( modeLabel );
154        QString modeText;
155        switch ( choice )
156        {
157        case ChoicePage::Alongside:
158            modeText = tr( "Install %1 <strong>alongside</strong> another operating system." )
159                       .arg( *Calamares::Branding::ShortVersionedName );
160            break;
161        case ChoicePage::Erase:
162            modeText = tr( "<strong>Erase</strong> disk and install %1." )
163                       .arg( *Calamares::Branding::ShortVersionedName );
164            break;
165        case ChoicePage::Replace:
166            modeText = tr( "<strong>Replace</strong> a partition with %1." )
167                       .arg( *Calamares::Branding::ShortVersionedName );
168            break;
169        case ChoicePage::NoChoice:
170        case ChoicePage::Manual:
171            modeText = tr( "<strong>Manual</strong> partitioning." );
172        }
173        modeLabel->setText( modeText );
174    }
175    for ( const auto& info : list )
176    {
177        QLabel* diskInfoLabel = new QLabel;
178        if ( list.length() == 1 ) // this is the only disk preview
179        {
180            QString modeText;
181            switch ( choice )
182            {
183            case ChoicePage::Alongside:
184                modeText = tr( "Install %1 <strong>alongside</strong> another operating system on disk <strong>%2</strong> (%3)." )
185                           .arg( *Calamares::Branding::ShortVersionedName )
186                           .arg( info.deviceNode )
187                           .arg( info.deviceName );
188                break;
189            case ChoicePage::Erase:
190                modeText = tr( "<strong>Erase</strong> disk <strong>%2</strong> (%3) and install %1." )
191                           .arg( *Calamares::Branding::ShortVersionedName )
192                           .arg( info.deviceNode )
193                           .arg( info.deviceName );
194                break;
195            case ChoicePage::Replace:
196                modeText = tr( "<strong>Replace</strong> a partition on disk <strong>%2</strong> (%3) with %1." )
197                           .arg( *Calamares::Branding::ShortVersionedName )
198                           .arg( info.deviceNode )
199                           .arg( info.deviceName );
200                break;
201            case ChoicePage::NoChoice:
202            case ChoicePage::Manual:
203                modeText = tr( "<strong>Manual</strong> partitioning on disk <strong>%1</strong> (%2)." )
204                           .arg( info.deviceNode )
205                           .arg( info.deviceName );
206            }
207            diskInfoLabel->setText( modeText );
208        }
209        else // multiple disk previews!
210        {
211            diskInfoLabel->setText( tr( "Disk <strong>%1</strong> (%2)" )
212                                        .arg( info.deviceNode )
213                                        .arg( info.deviceName ) );
214        }
215        formLayout->addRow( diskInfoLabel );
216
217        PartitionBarsView* preview;
218        PartitionLabelsView* previewLabels;
219        QVBoxLayout* field;
220
221        PartitionBarsView::NestedPartitionsMode mode = Calamares::JobQueue::instance()->globalStorage()->
222                                                       value( "drawNestedPartitions" ).toBool() ?
223                                                           PartitionBarsView::DrawNestedPartitions :
224                                                           PartitionBarsView::NoNestedPartitions;
225        preview = new PartitionBarsView;
226        preview->setNestedPartitionsMode( mode );
227        previewLabels = new PartitionLabelsView;
228        previewLabels->setExtendedPartitionHidden( mode == PartitionBarsView::NoNestedPartitions );
229        preview->setModel( info.partitionModelBefore );
230        previewLabels->setModel( info.partitionModelBefore );
231        preview->setSelectionMode( QAbstractItemView::NoSelection );
232        previewLabels->setSelectionMode( QAbstractItemView::NoSelection );
233        info.partitionModelBefore->setParent( widget );
234        field = new QVBoxLayout;
235        CalamaresUtils::unmarginLayout( field );
236        field->setSpacing( 6 );
237        field->addWidget( preview );
238        field->addWidget( previewLabels );
239        formLayout->addRow( tr( "Current:" ), field );
240
241        preview = new PartitionBarsView;
242        preview->setNestedPartitionsMode( mode );
243        previewLabels = new PartitionLabelsView;
244        previewLabels->setExtendedPartitionHidden( mode == PartitionBarsView::NoNestedPartitions );
245        preview->setModel( info.partitionModelAfter );
246        previewLabels->setModel( info.partitionModelAfter );
247        preview->setSelectionMode( QAbstractItemView::NoSelection );
248        previewLabels->setSelectionMode( QAbstractItemView::NoSelection );
249        previewLabels->setCustomNewRootLabel( *Calamares::Branding::BootloaderEntryName );
250        info.partitionModelAfter->setParent( widget );
251        field = new QVBoxLayout;
252        CalamaresUtils::unmarginLayout( field );
253        field->setSpacing( 6 );
254        field->addWidget( preview );
255        field->addWidget( previewLabels );
256        formLayout->addRow( tr( "After:" ), field );
257    }
258    QStringList jobsLines;
259    foreach ( const Calamares::job_ptr& job, jobs() )
260    {
261        if ( !job->prettyDescription().isEmpty() )
262        jobsLines.append( job->prettyDescription() );
263    }
264    if ( !jobsLines.isEmpty() )
265    {
266        QLabel* jobsLabel = new QLabel( widget );
267        mainLayout->addWidget( jobsLabel );
268        jobsLabel->setText( jobsLines.join( "<br/>" ) );
269        jobsLabel->setMargin( CalamaresUtils::defaultFontHeight() / 2 );
270        QPalette pal;
271        pal.setColor( QPalette::Background, pal.background().color().lighter( 108 ) );
272        jobsLabel->setAutoFillBackground( true );
273        jobsLabel->setPalette( pal );
274    }
275    return widget;
276}
277
278
279void
280PartitionViewStep::next()
281{
282    if ( m_choicePage == m_widget->currentWidget() )
283    {
284        if ( m_choicePage->currentChoice() == ChoicePage::Manual )
285        {
286            m_widget->setCurrentWidget( m_manualPartitionPage );
287            if ( m_core->isDirty() )
288                m_manualPartitionPage->onRevertClicked();
289        }
290        else if ( m_choicePage->currentChoice() == ChoicePage::Erase )
291        {
292            emit done();
293            return;
294        }
295        else if ( m_choicePage->currentChoice() == ChoicePage::Alongside )
296        {
297            emit done();
298            return;
299        }
300        else if ( m_choicePage->currentChoice() == ChoicePage::Replace )
301        {
302            emit done();
303            return;
304        }
305        cDebug() << "Choice applied: " << m_choicePage->currentChoice();
306        return;
307    }
308    emit done();
309}
310
311
312void
313PartitionViewStep::back()
314{
315    if ( m_widget->currentWidget() != m_choicePage )
316        m_widget->setCurrentWidget( m_choicePage );
317}
318
319
320bool
321PartitionViewStep::isNextEnabled() const
322{
323    if ( m_choicePage && m_choicePage == m_widget->currentWidget() )
324        return m_choicePage->isNextEnabled();
325
326    if ( m_manualPartitionPage && m_manualPartitionPage == m_widget->currentWidget() )
327        return m_core->hasRootMountPoint();
328
329    return false;
330}
331
332
333bool
334PartitionViewStep::isBackEnabled() const
335{
336    return true;
337}
338
339
340bool
341PartitionViewStep::isAtBeginning() const
342{
343    if ( m_widget->currentWidget() == m_manualPartitionPage )
344        return false;
345    return true;
346}
347
348
349bool
350PartitionViewStep::isAtEnd() const
351{
352    if ( m_choicePage == m_widget->currentWidget() )
353    {
354        if ( m_choicePage->currentChoice() == ChoicePage::Erase ||
355             m_choicePage->currentChoice() == ChoicePage::Replace ||
356             m_choicePage->currentChoice() == ChoicePage::Alongside )
357            return true;
358        return false;
359    }
360    return true;
361}
362
363
364void
365PartitionViewStep::onActivate()
366{
367    // if we're coming back to PVS from the next VS
368    if ( m_widget->currentWidget() == m_choicePage &&
369         m_choicePage->currentChoice() == ChoicePage::Alongside )
370    {
371        m_choicePage->applyActionChoice( ChoicePage::Alongside );
372//        m_choicePage->reset();
373        //FIXME: ReplaceWidget should be reset maybe?
374    }
375}
376
377
378void
379PartitionViewStep::onLeave()
380{
381    if ( m_widget->currentWidget() == m_choicePage )
382    {
383        m_choicePage->onLeave();
384        return;
385    }
386
387    if ( m_widget->currentWidget() == m_manualPartitionPage )
388    {
389        if ( PartUtils::isEfiSystem() )
390        {
391            QString espMountPoint = Calamares::JobQueue::instance()->globalStorage()->
392                                        value( "efiSystemPartition").toString();
393            Partition* esp = m_core->findPartitionByMountPoint( espMountPoint );
394
395            QString message;
396            QString description;
397            if ( !esp )
398            {
399                message = tr( "No EFI system partition configured" );
400                description = tr( "An EFI system partition is necessary to start %1."
401                                  "<br/><br/>"
402                                  "To configure an EFI system partition, go back and "
403                                  "select or create a FAT32 filesystem with the "
404                                  "<strong>esp</strong> flag enabled and mount point "
405                                  "<strong>%2</strong>.<br/><br/>"
406                                  "You can continue without setting up an EFI system "
407                                  "partition but your system may fail to start." )
408                              .arg( *Calamares::Branding::ShortProductName )
409                              .arg( espMountPoint );
410            }
411            else if ( esp && !PartUtils::isEfiBootable( esp ) )
412            {
413                message = tr( "EFI system partition flag not set" );
414                description = tr( "An EFI system partition is necessary to start %1."
415                                  "<br/><br/>"
416                                  "A partition was configured with mount point "
417                                  "<strong>%2</strong> but its <strong>esp</strong> "
418                                  "flag is not set.<br/>"
419                                  "To set the flag, go back and edit the partition."
420                                  "<br/><br/>"
421                                  "You can continue without setting the flag but your "
422                                  "system may fail to start." )
423                              .arg( *Calamares::Branding::ShortProductName )
424                              .arg( espMountPoint );
425            }
426
427            if ( !message.isEmpty() )
428            {
429                cWarning() << message;
430                QMessageBox::warning( m_manualPartitionPage,
431                                      message,
432                                      description );
433            }
434        }
435
436        Partition* root_p = m_core->findPartitionByMountPoint( "/" );
437        Partition* boot_p = m_core->findPartitionByMountPoint( "/boot" );
438
439        if ( root_p and boot_p )
440        {
441            QString message;
442            QString description;
443
444            // If the root partition is encrypted, and there's a separate boot
445            // partition which is not encrypted
446            if ( root_p->fileSystem().type() == FileSystem::Luks &&
447                 boot_p->fileSystem().type() != FileSystem::Luks )
448            {
449                message = tr( "Boot partition not encrypted" );
450                description = tr( "A separate boot partition was set up together with "
451                                  "an encrypted root partition, but the boot partition "
452                                  "is not encrypted."
453                                  "<br/><br/>"
454                                  "There are security concerns with this kind of "
455                                  "setup, because important system files are kept "
456                                  "on an unencrypted partition.<br/>"
457                                  "You may continue if you wish, but filesystem "
458                                  "unlocking will happen later during system startup."
459                                  "<br/>To encrypt the boot partition, go back and "
460                                  "recreate it, selecting <strong>Encrypt</strong> "
461                                  "in the partition creation window." );
462
463                QMessageBox::warning( m_manualPartitionPage,
464                                      message,
465                                      description );
466            }
467        }
468    }
469}
470
471
472void
473PartitionViewStep::setConfigurationMap( const QVariantMap& configurationMap )
474{
475    // Copy the efiSystemPartition setting to the global storage. It is needed not only in
476    // the EraseDiskPage, but also in the bootloader configuration modules (grub, bootloader).
477    Calamares::GlobalStorage* gs = Calamares::JobQueue::instance()->globalStorage();
478    if ( configurationMap.contains( "efiSystemPartition" ) &&
479         configurationMap.value( "efiSystemPartition" ).type() == QVariant::String &&
480         !configurationMap.value( "efiSystemPartition" ).toString().isEmpty() )
481    {
482        gs->insert( "efiSystemPartition", configurationMap.value( "efiSystemPartition" ).toString() );
483    }
484    else
485    {
486        gs->insert( "efiSystemPartition", QStringLiteral( "/boot/efi" ) );
487    }
488
489    if ( configurationMap.contains( "ensureSuspendToDisk" ) &&
490         configurationMap.value( "ensureSuspendToDisk" ).type() == QVariant::Bool )
491    {
492        gs->insert( "ensureSuspendToDisk", configurationMap.value( "ensureSuspendToDisk" ).toBool() );
493    }
494    else
495    {
496        gs->insert( "ensureSuspendToDisk", true );
497    }
498
499    if ( configurationMap.contains( "neverCreateSwap" ) &&
500         configurationMap.value( "neverCreateSwap" ).type() == QVariant::Bool )
501    {
502        gs->insert( "neverCreateSwap", configurationMap.value( "neverCreateSwap" ).toBool() );
503    }
504    else
505    {
506        gs->insert( "neverCreateSwap", false );
507    }
508
509    if ( configurationMap.contains( "drawNestedPartitions" ) &&
510         configurationMap.value( "drawNestedPartitions" ).type() == QVariant::Bool )
511    {
512        gs->insert( "drawNestedPartitions",
513                    configurationMap.value( "drawNestedPartitions", false ).toBool() );
514    }
515    else
516    {
517        gs->insert( "drawNestedPartitions", false );
518    }
519
520    if ( configurationMap.contains( "alwaysShowPartitionLabels" ) &&
521         configurationMap.value( "alwaysShowPartitionLabels" ).type() == QVariant::Bool )
522    {
523        gs->insert( "alwaysShowPartitionLabels",
524                    configurationMap.value( "alwaysShowPartitionLabels", true ).toBool() );
525    }
526    else
527    {
528        gs->insert( "alwaysShowPartitionLabels", true );
529    }
530
531    if ( configurationMap.contains( "defaultFileSystemType" ) &&
532         configurationMap.value( "defaultFileSystemType" ).type() == QVariant::String &&
533         !configurationMap.value( "defaultFileSystemType" ).toString().isEmpty() )
534    {
535        QString typeString = configurationMap.value( "defaultFileSystemType" ).toString();
536        gs->insert( "defaultFileSystemType", typeString );
537        if ( FileSystem::typeForName( typeString ) == FileSystem::Unknown )
538        {
539            cWarning() << "bad default filesystem configuration for partition module. Reverting to ext4 as default.";
540            gs->insert( "defaultFileSystemType", "ext4" );
541        }
542    }
543    else
544    {
545        gs->insert( "defaultFileSystemType", QStringLiteral( "ext4" ) );
546    }
547
548    if ( configurationMap.contains( "enableLuksAutomatedPartitioning" ) &&
549         configurationMap.value( "enableLuksAutomatedPartitioning" ).type() == QVariant::Bool )
550    {
551        gs->insert( "enableLuksAutomatedPartitioning",
552                    configurationMap.value( "enableLuksAutomatedPartitioning" ).toBool() );
553    }
554    else
555    {
556        gs->insert( "enableLuksAutomatedPartitioning", true );
557    }
558
559
560    // Now that we have the config, we load the PartitionCoreModule in the background
561    // because it could take a while. Then when it's done, we can set up the widgets
562    // and remove the spinner.
563    QFutureWatcher< void >* watcher = new QFutureWatcher< void >();
564    connect( watcher, &QFutureWatcher< void >::finished,
565             this, [ this, watcher ]
566    {
567        continueLoading();
568        watcher->deleteLater();
569    } );
570
571    QFuture< void > future =
572            QtConcurrent::run( this, &PartitionViewStep::initPartitionCoreModule );
573    watcher->setFuture( future );
574}
575
576
577QList< Calamares::job_ptr >
578PartitionViewStep::jobs() const
579{
580    return m_core->jobs();
581}
582
583
584CALAMARES_PLUGIN_FACTORY_DEFINITION( PartitionViewStepFactory, registerPlugin<PartitionViewStep>(); )
Note: See TracBrowser for help on using the repository browser.