calamares/src/modules/partition/CreatePartitionDialog.cpp

209 lines
6.4 KiB
C++
Raw Normal View History

/* === This file is part of Calamares - <http://github.com/calamares> ===
*
* Copyright 2014, Aurélien Gâteau <agateau@kde.org>
*
* 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 <http://www.gnu.org/licenses/>.
*/
#include <CreatePartitionDialog.h>
#include <PartitionInfo.h>
#include <ui_CreatePartitionDialog.h>
2014-07-16 10:35:09 +02:00
#include <utils/Logger.h>
// CalaPM
#include <core/device.h>
#include <core/partition.h>
#include <fs/filesystem.h>
#include <fs/filesystemfactory.h>
2014-07-04 16:14:06 +02:00
// Qt
#include <QComboBox>
#include <QSet>
2014-07-04 16:14:06 +02:00
static QSet< FileSystem::Type > s_unmountableFS(
{
FileSystem::Unformatted,
FileSystem::LinuxSwap
} );
CreatePartitionDialog::CreatePartitionDialog( Device* device, PartitionNode* parentPartition, QWidget* parentWidget )
: QDialog( parentWidget )
, m_ui( new Ui_CreatePartitionDialog )
, m_device( device )
, m_parent( parentPartition )
{
m_ui->setupUi( this );
FileSystemFactory::init();
bool parentIsPartitionTable = parentPartition->isRoot();
// Partition types
QString fixedPartitionType;
if ( !parentIsPartitionTable )
{
m_role = PartitionRole( PartitionRole::Logical );
fixedPartitionType = tr( "Logical" );
}
else if ( m_device->partitionTable()->hasExtended() )
{
m_role = PartitionRole( PartitionRole::Primary );
fixedPartitionType = tr( "Primary" );
}
if ( fixedPartitionType.isEmpty() )
m_ui->fixedPartitionLabel->hide();
else
{
m_ui->fixedPartitionLabel->setText( fixedPartitionType );
m_ui->primaryRadioButton->hide();
m_ui->extendedRadioButton->hide();
}
// File system
QStringList fsNames;
for ( auto fs : FileSystemFactory::map() )
{
if ( fs->supportCreate() != FileSystem::cmdSupportNone && fs->type() != FileSystem::Extended )
fsNames << fs->name();
}
m_ui->fsComboBox->addItems( fsNames );
// Connections
connect( m_ui->fsComboBox, SIGNAL( activated( int ) ), SLOT( updateMountPointUi() ) );
connect( m_ui->extendedRadioButton, SIGNAL( toggled( bool ) ), SLOT( updateMountPointUi() ) );
}
CreatePartitionDialog::~CreatePartitionDialog()
{}
Partition*
CreatePartitionDialog::createPartition()
{
if ( m_role.roles() == PartitionRole::None )
{
m_role = PartitionRole(
2014-07-02 14:12:47 +02:00
m_ui->extendedRadioButton->isChecked()
? PartitionRole::Extended
: PartitionRole::Primary
);
}
2014-07-16 10:35:09 +02:00
qint64 lastSector;
int mbSize = m_ui->sizeSpinBox->value();
2014-07-16 10:36:17 +02:00
if ( mbSize == m_ui->sizeSpinBox->maximum() )
{
2014-07-16 10:35:09 +02:00
// If we are at the maximum value, select the last sector to avoid
// potential rounding errors which could leave a few sectors at the end
// unused
lastSector = m_maxSector;
2014-07-16 10:36:17 +02:00
}
else
{
2014-07-16 10:35:09 +02:00
lastSector = m_minSector + qint64( mbSize ) * 1024 * 1024 / m_device->logicalSectorSize();
Q_ASSERT( lastSector <= m_maxSector );
if ( lastSector > m_maxSector )
{
cDebug() << "lastSector (" << lastSector << ") > m_maxSector (" << m_maxSector << "). This should not happen!";
lastSector = m_maxSector;
}
}
FileSystem::Type type = m_role.has( PartitionRole::Extended )
2014-07-02 14:12:47 +02:00
? FileSystem::Extended
: FileSystem::typeForName( m_ui->fsComboBox->currentText() );
2014-07-16 10:35:09 +02:00
FileSystem* fs = FileSystemFactory::create( type, m_minSector, lastSector );
auto partition = new Partition(
m_parent,
*m_device,
m_role,
2014-07-16 10:35:09 +02:00
fs, m_minSector, lastSector,
QString() /* path */,
PartitionTable::FlagNone /* availableFlags */,
2014-07-04 16:34:30 +02:00
QString() /* mountPoint */,
false /* mounted */,
PartitionTable::FlagNone /* activeFlags */,
Partition::StateNew
);
PartitionInfo::setMountPoint( partition, m_ui->mountPointComboBox->currentText() );
PartitionInfo::setFormat( partition, true );
return partition;
}
void
CreatePartitionDialog::updateMountPointUi()
{
bool enabled = m_ui->primaryRadioButton->isChecked();
if ( enabled )
{
FileSystem::Type type = FileSystem::typeForName( m_ui->fsComboBox->currentText() );
enabled = !s_unmountableFS.contains( type );
}
m_ui->mountPointLabel->setEnabled( enabled );
m_ui->mountPointComboBox->setEnabled( enabled );
}
void
CreatePartitionDialog::initSectorRange( Partition* partition )
{
PartitionTable* table = m_device->partitionTable();
m_minSector = partition->firstSector() - table->freeSectorsBefore( *partition );
m_maxSector = partition->lastSector() + table->freeSectorsAfter( *partition );
2014-07-16 10:34:56 +02:00
m_ui->sizeSpinBox->setMaximum( mbSizeForSectorRange( m_minSector, m_maxSector ) );
m_ui->sizeSpinBox->setValue( m_ui->sizeSpinBox->maximum() );
}
void
CreatePartitionDialog::initFromFreeSpace( Partition* freeSpacePartition )
{
initSectorRange( freeSpacePartition );
}
void
CreatePartitionDialog::initFromPartitionToCreate( Partition* partition )
{
Q_ASSERT( partition );
bool isExtended = partition->roles().has( PartitionRole::Extended );
Q_ASSERT( !isExtended );
if ( isExtended )
{
cDebug() << "Editing extended partitions is not supported for now";
return;
}
initSectorRange( partition );
// Size
2014-07-16 10:34:56 +02:00
m_ui->sizeSpinBox->setValue( mbSizeForSectorRange( partition->firstSector(), partition->lastSector() ) );
// File System
FileSystem::Type fsType = partition->fileSystem().type();
m_ui->fsComboBox->setCurrentText( FileSystem::nameForType( fsType ) );
// Mount point
m_ui->mountPointComboBox->setCurrentText( PartitionInfo::mountPoint( partition ) );
updateMountPointUi();
}
2014-07-16 10:34:56 +02:00
qint64
CreatePartitionDialog::mbSizeForSectorRange( qint64 first, qint64 last ) const
{
return ( last - first + 1 ) * m_device->logicalSectorSize() / 1024 / 1024;
}