/* === This file is part of Calamares - === * * Copyright 2014-2015, Teo Mrnjavac * Copyright 2014, Aurélien Gâteau * * Calamares is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * Calamares is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with Calamares. If not, see . */ #include "ReplacePage.h" #include "ui_ReplacePage.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include ReplacePage::ReplacePage( PartitionCoreModule* core, QWidget* parent ) : QWidget( parent ) , m_ui( new Ui_ReplacePage ) , m_core( core ) , m_isEfi( false ) { m_ui->setupUi( this ); m_ui->deviceComboBox->setModel( m_core->deviceModel() ); m_ui->partitionPreview->setLabelsVisible( true ); loadEfiSystemPartitions(); // updateButtons(); updateFromCurrentDevice(); connect( m_ui->deviceComboBox, &QComboBox::currentTextChanged, [ this ]( const QString& /* text */ ) { updateFromCurrentDevice(); } ); CALAMARES_RETRANSLATE( m_ui->retranslateUi( this ); onPartitionSelected(); ) } ReplacePage::~ReplacePage() {} bool ReplacePage::isNextEnabled() const { return m_nextEnabled; } void ReplacePage::reset() { int oldDeviceIndex = m_ui->deviceComboBox->currentIndex(); m_core->revert(); loadEfiSystemPartitions(); m_ui->deviceComboBox->setCurrentIndex( oldDeviceIndex ); updateFromCurrentDevice(); } void ReplacePage::applyChanges() { PartitionModel* model = qobject_cast< PartitionModel* >( m_ui->partitionTreeView->model() ); if ( model ) { Partition* partition = model->partitionForIndex( m_ui->partitionTreeView->currentIndex() ); if ( partition ) { Device* dev = model->device(); Partition* newPartition = PMUtils::createNewPartition( partition->parent(), *dev, partition->roles(), FileSystem::Ext4, partition->firstSector(), partition->lastSector() ); PartitionInfo::setMountPoint( newPartition, "/" ); PartitionInfo::setFormat( newPartition, true ); m_core->deletePartition( dev, partition ); m_core->createPartition( dev, newPartition ); if ( m_isEfi ) { if ( m_efiSystemPartitions.count() == 1 ) { PartitionInfo::setMountPoint( m_efiSystemPartitions.first(), Calamares::JobQueue::instance()-> globalStorage()-> value( "efiSystemPartition" ).toString() ); } else if ( m_efiSystemPartitions.count() > 1 ) { PartitionInfo::setMountPoint( m_efiSystemPartitions.at( m_ui->bootComboBox->currentIndex() ), Calamares::JobQueue::instance()-> globalStorage()-> value( "efiSystemPartition" ).toString() ); } } m_core->dumpQueue(); } } } void ReplacePage::onPartitionSelected() { if ( Calamares::JobQueue::instance()->globalStorage()->value( "firmwareType" ) == "efi" ) m_isEfi = true; cDebug() << "Partition selected in Replace page."; if ( m_ui->partitionTreeView->currentIndex() == QModelIndex() ) { updateStatus( CalamaresUtils::PartitionPartition, tr( "Select where to install %1.
" "Warning: this will delete all files " "on the selected partition." ) .arg( Calamares::Branding::instance()-> string( Calamares::Branding::VersionedName ) ) ); setNextEnabled( false ); return; } bool ok = false; double requiredSpaceB = Calamares::JobQueue::instance() ->globalStorage() ->value( "requiredStorageGB" ) .toDouble( &ok ) * 1024 * 1024 * 1024; PartitionModel* model = qobject_cast< PartitionModel* >( m_ui->partitionTreeView->model() ); if ( model && ok ) { QStringList osproberLines = Calamares::JobQueue::instance() ->globalStorage() ->value( "osproberLines" ).toStringList(); Partition* partition = model->partitionForIndex( m_ui->partitionTreeView->currentIndex() ); if ( !partition || partition->state() != Partition::StateNone ) { updateStatus( CalamaresUtils::Fail, tr( "The selected item does not appear to be a valid partition." ) ); setNextEnabled( false ); return; } if ( partition->roles().has( PartitionRole::Unallocated ) ) { updateStatus( CalamaresUtils::Fail, tr( "%1 cannot be installed on empty space. Please select an " "existing partition." ) .arg( Calamares::Branding::instance()-> string( Calamares::Branding::VersionedName ) ) ); setNextEnabled( false ); return; } if ( partition->roles().has( PartitionRole::Extended ) ) { updateStatus( CalamaresUtils::Fail, tr( "%1 cannot be installed on an extended partition. Please select an " "existing primary or logical partition." ) .arg( Calamares::Branding::instance()-> string( Calamares::Branding::VersionedName ) ) ); setNextEnabled( false ); return; } if ( partition->partitionPath().isEmpty() ) { updateStatus( CalamaresUtils::Fail, tr( "%1 cannot be installed on this partition." ) .arg( Calamares::Branding::instance()-> string( Calamares::Branding::VersionedName ) ) ); setNextEnabled( false ); return; } QString prettyName = tr( "Data partition (%1)" ) .arg( partition->fileSystem().name() ); foreach ( const QString& line, osproberLines ) { QStringList lineColumns = line.split( ':' ); QString path = lineColumns.value( 0 ).simplified(); if ( path == partition->partitionPath() ) { QString osName; if ( !lineColumns.value( 1 ).simplified().isEmpty() ) osName = lineColumns.value( 1 ).simplified(); else if ( !lineColumns.value( 2 ).simplified().isEmpty() ) osName = lineColumns.value( 2 ).simplified(); if ( osName.isEmpty() ) { prettyName = tr( "Unknown system partition (%1)" ) .arg( partition->fileSystem().name() ); } else { prettyName = tr ( "%1 system partition (%2)" ) .arg( osName.replace( 0, 1, osName.at( 0 ).toUpper() ) ) .arg( partition->fileSystem().name() ); } break; } } if ( partition->capacity() < requiredSpaceB ) { updateStatus( CalamaresUtils::Fail, tr( "%4

" "The partition %1 is too small for %2. Please select a partition " "with capacity at least %3 GiB." ) .arg( partition->partitionPath() ) .arg( Calamares::Branding::instance()-> string( Calamares::Branding::VersionedName ) ) .arg( requiredSpaceB / ( 1024. * 1024. * 1024. ), 0, 'f', 1 ) .arg( prettyName ) ); setNextEnabled( false ); return; } m_ui->bootComboBox->hide(); m_ui->bootComboBox->clear(); m_ui->bootStatusLabel->hide(); m_ui->bootStatusLabel->clear(); cDebug() << "isEfi:" << m_isEfi; if ( m_isEfi ) { cDebug() << "Assuming there should be an EFI System Partition somewhere..."; cDebug() << "m_efiSystemPartitions.count() is" << m_efiSystemPartitions.count(); if ( m_efiSystemPartitions.count() == 0 ) { updateStatus( CalamaresUtils::Fail, tr( "%2

" "An EFI system partition cannot be found anywhere " "on this system. Please go back and use manual " "partitioning to set up %1." ) .arg( Calamares::Branding::instance()-> string( Calamares::Branding::ShortProductName ) ) .arg( prettyName ) ); setNextEnabled( false ); } else if ( m_efiSystemPartitions.count() == 1 ) { updateStatus( CalamaresUtils::PartitionPartition, tr( "%3

" "%1 will be installed on %2.
" "Warning: all data on partition " "%2 will be lost.") .arg( Calamares::Branding::instance()-> string( Calamares::Branding::VersionedName ) ) .arg( partition->partitionPath() ) .arg( prettyName ) ); m_ui->bootStatusLabel->show(); m_ui->bootStatusLabel->setText( tr( "The EFI system partition at %1 will be used for starting %2." ) .arg( m_efiSystemPartitions.first()->partitionPath() ) .arg( Calamares::Branding::instance()-> string( Calamares::Branding::ShortProductName ) ) ); setNextEnabled( true ); } else { updateStatus( CalamaresUtils::PartitionPartition, tr( "%3

" "%1 will be installed on %2.
" "Warning: all data on partition " "%2 will be lost.") .arg( Calamares::Branding::instance()-> string( Calamares::Branding::VersionedName ) ) .arg( partition->partitionPath() ) .arg( prettyName ) ); m_ui->bootStatusLabel->show(); m_ui->bootStatusLabel->setText( tr( "EFI system partition:" ) ); m_ui->bootComboBox->show(); for ( int i = 0; i < m_efiSystemPartitions.count(); ++i ) { Partition* efiPartition = m_efiSystemPartitions.at( i ); m_ui->bootComboBox->addItem( efiPartition->partitionPath(), i ); if ( efiPartition->devicePath() == partition->devicePath() && efiPartition->number() == 1 ) m_ui->bootComboBox->setCurrentIndex( i ); } setNextEnabled( true ); } } else { cDebug() << "System is NOT EFI, bootloader will be installed on MBR."; updateStatus( CalamaresUtils::PartitionPartition, tr( "%3

" "%1 will be installed on %2.
" "Warning: all data on partition " "%2 will be lost.") .arg( Calamares::Branding::instance()-> string( Calamares::Branding::VersionedName ) ) .arg( partition->partitionPath() ) .arg( prettyName ) ); setNextEnabled( true ); } } } void ReplacePage::setNextEnabled( bool enabled ) { if ( enabled == m_nextEnabled ) return; m_nextEnabled = enabled; emit nextStatusChanged( enabled ); } void ReplacePage::updateStatus( CalamaresUtils::ImageType imageType, const QString& text ) { int iconSize = CalamaresUtils::defaultFontHeight() * 8; m_ui->selectedIconLabel->setPixmap( CalamaresUtils::defaultPixmap( imageType, CalamaresUtils::Original, QSize( iconSize, iconSize ) ) ); m_ui->selectedIconLabel->setFixedHeight( iconSize ); m_ui->selectedStatusLabel->setText( text ); } void ReplacePage::updateFromCurrentDevice() { QModelIndex index = m_core->deviceModel()->index( m_ui->deviceComboBox->currentIndex(), 0 ); if ( !index.isValid() ) return; Device* device = m_core->deviceModel()->deviceForIndex( index ); QAbstractItemModel* oldModel = m_ui->partitionTreeView->model(); if ( oldModel ) disconnect( oldModel, 0, this, 0 ); PartitionModel* model = m_core->partitionModelForDevice( device ); m_ui->partitionPreview->setModel( model ); m_ui->partitionTreeView->setModel( model ); m_ui->partitionTreeView->expandAll(); // Must be done here because we need to have a model set to define // individual column resize mode QHeaderView* header = m_ui->partitionTreeView->header(); header->setSectionResizeMode( QHeaderView::ResizeToContents ); header->setSectionResizeMode( 0, QHeaderView::Stretch ); //updateButtons(); // Establish connection here because selection model is destroyed when // model changes connect( m_ui->partitionTreeView->selectionModel(), &QItemSelectionModel::currentRowChanged, this, &ReplacePage::onPartitionViewActivated ); connect( model, &QAbstractItemModel::modelReset, this, &ReplacePage::onPartitionModelReset ); } void ReplacePage::onPartitionViewActivated() { QModelIndex index = m_ui->partitionTreeView->currentIndex(); if ( !index.isValid() ) return; const PartitionModel* model = static_cast< const PartitionModel* >( index.model() ); Q_ASSERT( model ); Partition* partition = model->partitionForIndex( index ); Q_ASSERT( partition ); onPartitionSelected(); } void ReplacePage::onPartitionModelReset() { m_ui->partitionTreeView->expandAll(); onPartitionSelected(); } void ReplacePage::loadEfiSystemPartitions() { cDebug() << "Searching for partitions of type EFI System Partition on all disks."; m_efiSystemPartitions.clear(); m_ui->bootComboBox->hide(); m_ui->bootComboBox->clear(); m_ui->bootStatusLabel->hide(); m_ui->bootStatusLabel->clear(); QList< Device* > devices; for ( int row = 0; row < m_core->deviceModel()->rowCount(); ++row ) { Device* device = m_core->deviceModel()->deviceForIndex( m_core->deviceModel()->index( row ) ); devices.append( device ); cDebug() << "Will look in device" << device->deviceNode(); } //FIXME: Unfortunately right now we have to call sgdisk manually because // the KPM submodule does not expose the ESP flag from libparted. // The following findPartitions call and lambda should be scrapped and // rewritten based on libKPM. -- Teo 5/2015 m_efiSystemPartitions = PMUtils::findPartitions( devices, []( Partition* partition ) -> bool { QProcess process; cDebug() << "sgdisk checking partition type for" << partition->partitionPath(); process.setProgram( "sgdisk" ); process.setArguments( { "-i", QString::number( partition->number() ), partition->devicePath() } ); process.setProcessChannelMode( QProcess::MergedChannels ); process.start(); if ( process.waitForFinished() ) { if ( process.readAllStandardOutput() .contains( "C12A7328-F81F-11D2-BA4B-00A0C93EC93B" ) ) { cDebug() << "SUCCESS! Adding" << partition->partitionPath() << "to efiSystemPartitions list."; return true; } } return false; } ); }