calamares/src/modules/partition/CreatePartitionDialog.cpp

191 lines
6.0 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
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()
{}
PartitionInfo*
CreatePartitionDialog::createPartitionInfo()
{
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
);
auto info = new PartitionInfo( partition );
info->mountPoint = m_ui->mountPointComboBox->currentText();
2014-07-04 18:00:25 +02:00
info->format = true;
return info;
}
void
CreatePartitionDialog::updateMountPointUi()
{
static QSet< FileSystem::Type > unmountableFS( { FileSystem::Unformatted, FileSystem::LinuxSwap } );
bool enabled = m_ui->primaryRadioButton->isChecked();
if ( enabled )
{
FileSystem::Type type = FileSystem::typeForName( m_ui->fsComboBox->currentText() );
enabled = !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::initFromPartitionInfo( PartitionInfo* partitionInfo )
{
Q_ASSERT( partitionInfo );
Partition* partition = partitionInfo->partition;
initSectorRange( partition );
2014-07-16 10:34:56 +02:00
m_ui->sizeSpinBox->setValue( mbSizeForSectorRange( partition->firstSector(), partition->lastSector() ) );
m_ui->mountPointComboBox->setCurrentText( partitionInfo->mountPoint );
}
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;
}