Compare commits
67 Commits
v3.1.10
...
3.1.x-stab
Author | SHA1 | Date | |
---|---|---|---|
![]() |
5ea2e1c2fe | ||
![]() |
c9ced8c387 | ||
![]() |
4f16d65a8b | ||
![]() |
17fd10cdb1 | ||
![]() |
faf21ce438 | ||
![]() |
be5ad6944e | ||
![]() |
c9928c99c8 | ||
![]() |
1340613ef5 | ||
![]() |
76e37402b3 | ||
![]() |
0c1453ff18 | ||
![]() |
eea421f499 | ||
![]() |
6b7c8a694a | ||
![]() |
5b98e58ae7 | ||
![]() |
939cdff93b | ||
![]() |
aaae1507cd | ||
![]() |
3636226425 | ||
![]() |
fe98b789f0 | ||
![]() |
d5623af8ef | ||
![]() |
c0d5a153d4 | ||
![]() |
445f181cc3 | ||
![]() |
06e43a7312 | ||
![]() |
b8e6144553 | ||
![]() |
4c04260b97 | ||
![]() |
36aede52ef | ||
![]() |
fd1279dbe3 | ||
![]() |
9cdb6734bf | ||
![]() |
32a1c84935 | ||
![]() |
fb93a8288e | ||
![]() |
b0828faadb | ||
![]() |
c2efae765d | ||
![]() |
11652c5856 | ||
![]() |
94000b6847 | ||
![]() |
b3f0932ff9 | ||
![]() |
41e6f0e06c | ||
![]() |
5593be125f | ||
![]() |
661789825a | ||
![]() |
2e1f389997 | ||
![]() |
c2ee0c6ed4 | ||
![]() |
d42210eb93 | ||
![]() |
d5b46dfb86 | ||
![]() |
f2aeecf546 | ||
![]() |
10e71bab30 | ||
![]() |
e73d54767d | ||
![]() |
cf39dddbf3 | ||
![]() |
3f258d4bd9 | ||
![]() |
2db485bb33 | ||
![]() |
244919d6fe | ||
![]() |
8b3f71af40 | ||
![]() |
755c0cba18 | ||
![]() |
11fc3e0507 | ||
![]() |
afbdcc3782 | ||
![]() |
748ccf94e9 | ||
![]() |
6bd8c67ca9 | ||
![]() |
ad69eda337 | ||
![]() |
98b9f67e39 | ||
![]() |
02dfe51d55 | ||
![]() |
e628ddfdbf | ||
![]() |
b10b19e9ee | ||
![]() |
eb92755b0a | ||
![]() |
71966b5330 | ||
![]() |
1de2e94fd0 | ||
![]() |
fe8ff3ab05 | ||
![]() |
4e2e55a935 | ||
![]() |
ac92d4911d | ||
![]() |
3a94f02547 | ||
![]() |
cddc4699aa | ||
![]() |
1f49f764a6 |
@@ -114,6 +114,16 @@ find_package( Qt5 ${QT_VERSION} CONFIG REQUIRED Core Gui Widgets LinguistTools S
|
|||||||
find_package( YAMLCPP 0.5.1 REQUIRED )
|
find_package( YAMLCPP 0.5.1 REQUIRED )
|
||||||
find_package( PolkitQt5-1 REQUIRED )
|
find_package( PolkitQt5-1 REQUIRED )
|
||||||
|
|
||||||
|
# Find ECM once, and add it to the module search path; Calamares
|
||||||
|
# modules that need ECM can do
|
||||||
|
# find_package(ECM ${ECM_VERSION} REQUIRED NO_MODULE),
|
||||||
|
# no need to mess with the module path after.
|
||||||
|
set( ECM_VERSION 5.10.0 )
|
||||||
|
find_package(ECM ${ECM_VERSION} NO_MODULE)
|
||||||
|
if( ECM_FOUND )
|
||||||
|
set(CMAKE_MODULE_PATH ${ECM_MODULE_PATH} ${CMAKE_MODULE_PATH})
|
||||||
|
endif()
|
||||||
|
|
||||||
option( INSTALL_CONFIG "Install configuration files" ON )
|
option( INSTALL_CONFIG "Install configuration files" ON )
|
||||||
option( WITH_PYTHON "Enable Python modules API (requires Boost.Python)." ON )
|
option( WITH_PYTHON "Enable Python modules API (requires Boost.Python)." ON )
|
||||||
option( WITH_PYTHONQT "Enable next generation Python modules API (experimental, requires PythonQt)." OFF )
|
option( WITH_PYTHONQT "Enable next generation Python modules API (experimental, requires PythonQt)." OFF )
|
||||||
@@ -166,7 +176,7 @@ set( CALAMARES_TRANSLATION_LANGUAGES ar ast bg ca cs_CZ da de el en en_GB es_MX
|
|||||||
### Bump version here
|
### Bump version here
|
||||||
set( CALAMARES_VERSION_MAJOR 3 )
|
set( CALAMARES_VERSION_MAJOR 3 )
|
||||||
set( CALAMARES_VERSION_MINOR 1 )
|
set( CALAMARES_VERSION_MINOR 1 )
|
||||||
set( CALAMARES_VERSION_PATCH 10 )
|
set( CALAMARES_VERSION_PATCH 13 )
|
||||||
set( CALAMARES_VERSION_RC 0 )
|
set( CALAMARES_VERSION_RC 0 )
|
||||||
|
|
||||||
set( CALAMARES_VERSION ${CALAMARES_VERSION_MAJOR}.${CALAMARES_VERSION_MINOR}.${CALAMARES_VERSION_PATCH} )
|
set( CALAMARES_VERSION ${CALAMARES_VERSION_MAJOR}.${CALAMARES_VERSION_MINOR}.${CALAMARES_VERSION_PATCH} )
|
||||||
|
@@ -3,15 +3,27 @@ include( CalamaresAddTranslations )
|
|||||||
|
|
||||||
set( MODULE_DATA_DESTINATION share/calamares/modules )
|
set( MODULE_DATA_DESTINATION share/calamares/modules )
|
||||||
|
|
||||||
|
# Convenience function to indicate that a module has been skipped
|
||||||
|
# (optionally also why). Call this in the module's CMakeLists.txt
|
||||||
|
macro( calamares_skip_module )
|
||||||
|
set( SKIPPED_MODULES ${SKIPPED_MODULES} ${ARGV} PARENT_SCOPE )
|
||||||
|
endmacro()
|
||||||
|
|
||||||
function( calamares_add_module_subdirectory )
|
function( calamares_add_module_subdirectory )
|
||||||
set( SUBDIRECTORY ${ARGV0} )
|
set( SUBDIRECTORY ${ARGV0} )
|
||||||
|
|
||||||
|
set( SKIPPED_MODULES )
|
||||||
set( MODULE_CONFIG_FILES "" )
|
set( MODULE_CONFIG_FILES "" )
|
||||||
|
|
||||||
# If this subdirectory has a CMakeLists.txt, we add_subdirectory it...
|
# If this subdirectory has a CMakeLists.txt, we add_subdirectory it...
|
||||||
if( EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/${SUBDIRECTORY}/CMakeLists.txt" )
|
if( EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/${SUBDIRECTORY}/CMakeLists.txt" )
|
||||||
add_subdirectory( ${SUBDIRECTORY} )
|
add_subdirectory( ${SUBDIRECTORY} )
|
||||||
file( GLOB MODULE_CONFIG_FILES RELATIVE ${CMAKE_CURRENT_SOURCE_DIR}/${SUBDIRECTORY} "${SUBDIRECTORY}/*.conf" )
|
file( GLOB MODULE_CONFIG_FILES RELATIVE ${CMAKE_CURRENT_SOURCE_DIR}/${SUBDIRECTORY} "${SUBDIRECTORY}/*.conf" )
|
||||||
|
# Module has indicated it should be skipped, show that in
|
||||||
|
# the calling CMakeLists (which is src/modules/CMakeLists.txt normally).
|
||||||
|
if ( SKIPPED_MODULES )
|
||||||
|
set( SKIPPED_MODULES ${SKIPPED_MODULES} PARENT_SCOPE )
|
||||||
|
endif()
|
||||||
# ...otherwise, we look for a module.desc.
|
# ...otherwise, we look for a module.desc.
|
||||||
elseif( EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/${SUBDIRECTORY}/module.desc" )
|
elseif( EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/${SUBDIRECTORY}/module.desc" )
|
||||||
set( MODULES_DIR ${CMAKE_INSTALL_LIBDIR}/calamares/modules )
|
set( MODULES_DIR ${CMAKE_INSTALL_LIBDIR}/calamares/modules )
|
||||||
|
@@ -1,2 +1,2 @@
|
|||||||
FROM kdeneon/all
|
FROM kdeneon/all:user
|
||||||
RUN sudo apt-get update && sudo apt-get -y install build-essential cmake extra-cmake-modules gettext kio-dev libatasmart-dev libboost-python-dev libkf5config-dev libkf5coreaddons-dev libkf5i18n-dev libkf5iconthemes-dev libkf5parts-dev libkf5service-dev libkf5solid-dev libkpmcore-dev libparted-dev libpolkit-qt5-1-dev libqt5svg5-dev libqt5webkit5-dev libyaml-cpp-dev os-prober pkg-config python3-dev qtbase5-dev qtdeclarative5-dev qttools5-dev qttools5-dev-tools
|
RUN sudo apt-get update && sudo apt-get -y install build-essential cmake extra-cmake-modules gettext kio-dev libatasmart-dev libboost-python-dev libkf5config-dev libkf5coreaddons-dev libkf5i18n-dev libkf5iconthemes-dev libkf5parts-dev libkf5service-dev libkf5solid-dev libkpmcore-dev libparted-dev libpolkit-qt5-1-dev libqt5svg5-dev libqt5webkit5-dev libyaml-cpp-dev os-prober pkg-config python3-dev qtbase5-dev qtdeclarative5-dev qttools5-dev qttools5-dev-tools
|
||||||
|
@@ -55,14 +55,7 @@ CalamaresApplication::CalamaresApplication( int& argc, char* argv[] )
|
|||||||
|
|
||||||
QFont f = font();
|
QFont f = font();
|
||||||
|
|
||||||
cDebug() << "Default font ====="
|
cDebug() << "Default font size" << f.pointSize() << ';' << f.pixelSize() << "px";
|
||||||
<< "\nPixel size: " << f.pixelSize()
|
|
||||||
<< "\nPoint size: " << f.pointSize()
|
|
||||||
<< "\nPoint sizeF: " << f.pointSizeF()
|
|
||||||
<< "\nFont family: " << f.family()
|
|
||||||
<< "\nMetric height:" << QFontMetrics( f ).height();
|
|
||||||
// The following line blocks for 15s on Qt 5.1.0
|
|
||||||
cDebug() << "Font height:" << QFontMetrics( f ).height();
|
|
||||||
CalamaresUtils::setDefaultFontSize( f.pointSize() );
|
CalamaresUtils::setDefaultFontSize( f.pointSize() );
|
||||||
|
|
||||||
cDebug() << "Available languages:" << QString( CALAMARES_TRANSLATION_LANGUAGES ).split( ';' );
|
cDebug() << "Available languages:" << QString( CALAMARES_TRANSLATION_LANGUAGES ).split( ';' );
|
||||||
|
@@ -1,7 +1,7 @@
|
|||||||
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
*
|
*
|
||||||
* Copyright 2014-2015, Teo Mrnjavac <teo@kde.org>
|
* Copyright 2014-2015, Teo Mrnjavac <teo@kde.org>
|
||||||
* Copyright 2017, Adriaan de Groot <groot@kde.org>
|
* Copyright 2017-2018, Adriaan de Groot <groot@kde.org>
|
||||||
*
|
*
|
||||||
* Calamares is free software: you can redistribute it and/or modify
|
* Calamares is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -30,6 +30,7 @@
|
|||||||
|
|
||||||
#include <QApplication>
|
#include <QApplication>
|
||||||
#include <QBoxLayout>
|
#include <QBoxLayout>
|
||||||
|
#include <QCloseEvent>
|
||||||
#include <QDesktopWidget>
|
#include <QDesktopWidget>
|
||||||
#include <QLabel>
|
#include <QLabel>
|
||||||
#include <QTreeView>
|
#include <QTreeView>
|
||||||
@@ -37,10 +38,8 @@
|
|||||||
CalamaresWindow::CalamaresWindow( QWidget* parent )
|
CalamaresWindow::CalamaresWindow( QWidget* parent )
|
||||||
: QWidget( parent )
|
: QWidget( parent )
|
||||||
, m_debugWindow( nullptr )
|
, m_debugWindow( nullptr )
|
||||||
|
, m_viewManager( nullptr )
|
||||||
{
|
{
|
||||||
// Hide close button
|
|
||||||
setWindowFlags( Qt::Window | Qt::WindowTitleHint | Qt::CustomizeWindowHint );
|
|
||||||
|
|
||||||
CALAMARES_RETRANSLATE(
|
CALAMARES_RETRANSLATE(
|
||||||
setWindowTitle( tr( "%1 Installer" )
|
setWindowTitle( tr( "%1 Installer" )
|
||||||
.arg( *Calamares::Branding::ProductName ) );
|
.arg( *Calamares::Branding::ProductName ) );
|
||||||
@@ -139,10 +138,10 @@ CalamaresWindow::CalamaresWindow( QWidget* parent )
|
|||||||
CalamaresUtils::unmarginLayout( sideLayout );
|
CalamaresUtils::unmarginLayout( sideLayout );
|
||||||
CalamaresUtils::unmarginLayout( mainLayout );
|
CalamaresUtils::unmarginLayout( mainLayout );
|
||||||
|
|
||||||
Calamares::ViewManager* vm = Calamares::ViewManager::instance( this );
|
m_viewManager = Calamares::ViewManager::instance( this );
|
||||||
connect( vm, &Calamares::ViewManager::enlarge, this, &CalamaresWindow::enlarge );
|
connect( m_viewManager, &Calamares::ViewManager::enlarge, this, &CalamaresWindow::enlarge );
|
||||||
|
|
||||||
mainLayout->addWidget( vm->centralWidget() );
|
mainLayout->addWidget( m_viewManager->centralWidget() );
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@@ -156,3 +155,15 @@ CalamaresWindow::enlarge( QSize enlarge )
|
|||||||
|
|
||||||
resize( w, h );
|
resize( w, h );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
CalamaresWindow::closeEvent( QCloseEvent* event )
|
||||||
|
{
|
||||||
|
if ( ( !m_viewManager ) || m_viewManager->confirmCancelInstallation() )
|
||||||
|
{
|
||||||
|
event->accept();
|
||||||
|
qApp->quit();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
event->ignore();
|
||||||
|
}
|
||||||
|
@@ -1,7 +1,7 @@
|
|||||||
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
*
|
*
|
||||||
* Copyright 2014-2015, Teo Mrnjavac <teo@kde.org>
|
* Copyright 2014-2015, Teo Mrnjavac <teo@kde.org>
|
||||||
* Copyright 2017, Adriaan de Groot <groot@kde.org>
|
* Copyright 2017-2018, Adriaan de Groot <groot@kde.org>
|
||||||
*
|
*
|
||||||
* Calamares is free software: you can redistribute it and/or modify
|
* Calamares is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -26,6 +26,7 @@
|
|||||||
namespace Calamares
|
namespace Calamares
|
||||||
{
|
{
|
||||||
class DebugWindow;
|
class DebugWindow;
|
||||||
|
class ViewManager;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -46,8 +47,12 @@ public slots:
|
|||||||
*/
|
*/
|
||||||
void enlarge( QSize enlarge );
|
void enlarge( QSize enlarge );
|
||||||
|
|
||||||
|
protected:
|
||||||
|
virtual void closeEvent( QCloseEvent* e ) override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QPointer< Calamares::DebugWindow > m_debugWindow;
|
QPointer< Calamares::DebugWindow > m_debugWindow; // Managed by self
|
||||||
|
Calamares::ViewManager* m_viewManager;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif //CALAMARESWINDOW_H
|
#endif //CALAMARESWINDOW_H
|
||||||
|
@@ -256,5 +256,10 @@ System::getTotalMemoryB()
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
System::doChroot() const
|
||||||
|
{
|
||||||
|
return m_doChroot;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
@@ -114,6 +114,8 @@ public:
|
|||||||
*/
|
*/
|
||||||
DLLEXPORT QPair<quint64, float> getTotalMemoryB();
|
DLLEXPORT QPair<quint64, float> getTotalMemoryB();
|
||||||
|
|
||||||
|
DLLEXPORT bool doChroot() const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
static System* s_instance;
|
static System* s_instance;
|
||||||
|
|
||||||
|
@@ -1,7 +1,7 @@
|
|||||||
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
*
|
*
|
||||||
* Copyright 2014-2015, Teo Mrnjavac <teo@kde.org>
|
* Copyright 2014-2015, Teo Mrnjavac <teo@kde.org>
|
||||||
* Copyright 2017, Adriaan de Groot <groot@kde.org>
|
* Copyright 2017-2018, Adriaan de Groot <groot@kde.org>
|
||||||
*
|
*
|
||||||
* Calamares is free software: you can redistribute it and/or modify
|
* Calamares is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -88,30 +88,8 @@ ViewManager::ViewManager( QObject* parent )
|
|||||||
connect( m_back, &QPushButton::clicked, this, &ViewManager::back );
|
connect( m_back, &QPushButton::clicked, this, &ViewManager::back );
|
||||||
m_back->setEnabled( false );
|
m_back->setEnabled( false );
|
||||||
|
|
||||||
connect( m_quit, &QPushButton::clicked,
|
connect( m_quit, &QPushButton::clicked, this,
|
||||||
this, [this]()
|
[this]() { if ( this->confirmCancelInstallation() ) qApp->quit(); } );
|
||||||
{
|
|
||||||
// If it's NOT the last page of the last step, we ask for confirmation
|
|
||||||
if ( !( m_currentStep == m_steps.count() -1 &&
|
|
||||||
m_steps.last()->isAtEnd() ) )
|
|
||||||
{
|
|
||||||
QMessageBox mb( QMessageBox::Question,
|
|
||||||
tr( "Cancel installation?" ),
|
|
||||||
tr( "Do you really want to cancel the current install process?\n"
|
|
||||||
"The installer will quit and all changes will be lost." ),
|
|
||||||
QMessageBox::Yes | QMessageBox::No,
|
|
||||||
m_widget );
|
|
||||||
mb.setDefaultButton( QMessageBox::No );
|
|
||||||
mb.button( QMessageBox::Yes )->setText( tr( "&Yes" ) );
|
|
||||||
mb.button( QMessageBox::No )->setText( tr( "&No" ) );
|
|
||||||
int response = mb.exec();
|
|
||||||
if ( response == QMessageBox::Yes )
|
|
||||||
qApp->quit();
|
|
||||||
}
|
|
||||||
else // Means we're at the end, no need to confirm.
|
|
||||||
qApp->quit();
|
|
||||||
} );
|
|
||||||
|
|
||||||
connect( JobQueue::instance(), &JobQueue::failed,
|
connect( JobQueue::instance(), &JobQueue::failed,
|
||||||
this, &ViewManager::onInstallationFailed );
|
this, &ViewManager::onInstallationFailed );
|
||||||
connect( JobQueue::instance(), &JobQueue::finished,
|
connect( JobQueue::instance(), &JobQueue::finished,
|
||||||
@@ -302,4 +280,26 @@ ViewManager::back()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool ViewManager::confirmCancelInstallation()
|
||||||
|
{
|
||||||
|
// If it's NOT the last page of the last step, we ask for confirmation
|
||||||
|
if ( !( m_currentStep == m_steps.count() -1 &&
|
||||||
|
m_steps.last()->isAtEnd() ) )
|
||||||
|
{
|
||||||
|
QMessageBox mb( QMessageBox::Question,
|
||||||
|
tr( "Cancel installation?" ),
|
||||||
|
tr( "Do you really want to cancel the current install process?\n"
|
||||||
|
"The installer will quit and all changes will be lost." ),
|
||||||
|
QMessageBox::Yes | QMessageBox::No,
|
||||||
|
m_widget );
|
||||||
|
mb.setDefaultButton( QMessageBox::No );
|
||||||
|
mb.button( QMessageBox::Yes )->setText( tr( "&Yes" ) );
|
||||||
|
mb.button( QMessageBox::No )->setText( tr( "&No" ) );
|
||||||
|
int response = mb.exec();
|
||||||
|
return response == QMessageBox::Yes;
|
||||||
|
}
|
||||||
|
else // Means we're at the end, no need to confirm.
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
@@ -1,7 +1,7 @@
|
|||||||
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
*
|
*
|
||||||
* Copyright 2014-2015, Teo Mrnjavac <teo@kde.org>
|
* Copyright 2014-2015, Teo Mrnjavac <teo@kde.org>
|
||||||
* Copyright 2017, Adriaan de Groot <groot@kde.org>
|
* Copyright 2017-2018, Adriaan de Groot <groot@kde.org>
|
||||||
*
|
*
|
||||||
* Calamares is free software: you can redistribute it and/or modify
|
* Calamares is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -86,6 +86,13 @@ public:
|
|||||||
*/
|
*/
|
||||||
int currentStepIndex() const;
|
int currentStepIndex() const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @ brief Called when "Cancel" is clicked; asks for confirmation.
|
||||||
|
* Other means of closing Calamares also call this method, e.g. alt-F4.
|
||||||
|
* At the end of installation, no confirmation is asked. Returns true
|
||||||
|
* if the user confirms closing the window.
|
||||||
|
*/
|
||||||
|
bool confirmCancelInstallation();
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
/**
|
/**
|
||||||
|
@@ -1,5 +1,11 @@
|
|||||||
include( CMakeColors )
|
include( CMakeColors )
|
||||||
|
|
||||||
|
# The variable SKIP_MODULES can be set to skip particular modules;
|
||||||
|
# individual modules can also decide they must be skipped (e.g. OS-specific
|
||||||
|
# modules, or ones with unmet dependencies). Collect the skipped modules
|
||||||
|
# in this list.
|
||||||
|
set( LIST_SKIPPED_MODULES "" )
|
||||||
|
|
||||||
if( BUILD_TESTING )
|
if( BUILD_TESTING )
|
||||||
add_executable( test_conf test_conf.cpp )
|
add_executable( test_conf test_conf.cpp )
|
||||||
target_link_libraries( test_conf ${YAMLCPP_LIBRARY} )
|
target_link_libraries( test_conf ${YAMLCPP_LIBRARY} )
|
||||||
@@ -13,11 +19,24 @@ foreach( SUBDIRECTORY ${SUBDIRECTORIES} )
|
|||||||
if( NOT DO_SKIP EQUAL -1 )
|
if( NOT DO_SKIP EQUAL -1 )
|
||||||
message( "${ColorReset}-- Skipping module ${BoldRed}${SUBDIRECTORY}${ColorReset}." )
|
message( "${ColorReset}-- Skipping module ${BoldRed}${SUBDIRECTORY}${ColorReset}." )
|
||||||
message( "" )
|
message( "" )
|
||||||
|
list( APPEND LIST_SKIPPED_MODULES "${SUBDIRECTORY} (user request)" )
|
||||||
elseif( ( IS_DIRECTORY "${CMAKE_CURRENT_SOURCE_DIR}/${SUBDIRECTORY}" ) AND
|
elseif( ( IS_DIRECTORY "${CMAKE_CURRENT_SOURCE_DIR}/${SUBDIRECTORY}" ) AND
|
||||||
( DO_SKIP EQUAL -1 ) )
|
( DO_SKIP EQUAL -1 ) )
|
||||||
|
set( SKIPPED_MODULES )
|
||||||
calamares_add_module_subdirectory( ${SUBDIRECTORY} )
|
calamares_add_module_subdirectory( ${SUBDIRECTORY} )
|
||||||
|
if ( SKIPPED_MODULES )
|
||||||
|
list( APPEND LIST_SKIPPED_MODULES "${SKIPPED_MODULES}" )
|
||||||
|
endif()
|
||||||
endif()
|
endif()
|
||||||
endforeach()
|
endforeach()
|
||||||
|
|
||||||
|
if ( LIST_SKIPPED_MODULES )
|
||||||
|
message( "${ColorReset}-- Skipped modules:" )
|
||||||
|
foreach( SUBDIRECTORY ${LIST_SKIPPED_MODULES} )
|
||||||
|
message( "${ColorReset}-- Skipped ${BoldRed}${SUBDIRECTORY}${ColorReset}." )
|
||||||
|
endforeach()
|
||||||
|
message( "" )
|
||||||
|
endif()
|
||||||
|
|
||||||
include( CalamaresAddTranslations )
|
include( CalamaresAddTranslations )
|
||||||
add_calamares_python_translations( ${CALAMARES_TRANSLATION_LANGUAGES} )
|
add_calamares_python_translations( ${CALAMARES_TRANSLATION_LANGUAGES} )
|
||||||
|
@@ -5,7 +5,8 @@
|
|||||||
#
|
#
|
||||||
# Copyright 2014 - 2015, Philip Müller <philm@manjaro.org>
|
# Copyright 2014 - 2015, Philip Müller <philm@manjaro.org>
|
||||||
# Copyright 2014, Teo Mrnjavac <teo@kde.org>
|
# Copyright 2014, Teo Mrnjavac <teo@kde.org>
|
||||||
# Copyright 2017. Alf Gaida <agaida@siduction.org>
|
# Copyright 2017, Alf Gaida <agaida@siduction.org>
|
||||||
|
# Copyright 2017, Gabriel Craciunescu <crazy@frugalware.org>
|
||||||
#
|
#
|
||||||
# Calamares is free software: you can redistribute it and/or modify
|
# Calamares is free software: you can redistribute it and/or modify
|
||||||
# it under the terms of the GNU General Public License as published by
|
# it under the terms of the GNU General Public License as published by
|
||||||
@@ -20,26 +21,33 @@
|
|||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with Calamares. If not, see <http://www.gnu.org/licenses/>.
|
# along with Calamares. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
import subprocess
|
|
||||||
import shutil
|
|
||||||
|
|
||||||
import libcalamares
|
import libcalamares
|
||||||
|
|
||||||
|
|
||||||
def run():
|
def run():
|
||||||
"""
|
"""
|
||||||
Set hardware clock.
|
Set hardware clock.
|
||||||
"""
|
"""
|
||||||
|
hwclock_rtc = ["hwclock", "--systohc", "--utc"]
|
||||||
|
hwclock_isa = ["hwclock", "--systohc", "--utc", "--directisa"]
|
||||||
|
is_broken_rtc = False
|
||||||
|
is_broken_isa = False
|
||||||
|
|
||||||
root_mount_point = libcalamares.globalstorage.value("rootMountPoint")
|
ret = libcalamares.utils.target_env_call(hwclock_rtc)
|
||||||
try:
|
if ret != 0:
|
||||||
subprocess.check_call(["hwclock", "--systohc", "--utc"])
|
is_broken_rtc = True
|
||||||
except subprocess.CalledProcessError as e:
|
libcalamares.utils.debug("Hwclock returned error code {}".format(ret))
|
||||||
return (
|
libcalamares.utils.debug(" .. RTC method failed, trying ISA bus method.")
|
||||||
"Cannot set hardware clock.",
|
else:
|
||||||
"hwclock terminated with exit code {}.".format(e.returncode)
|
libcalamares.utils.debug("Hwclock set using RTC method.")
|
||||||
)
|
if is_broken_rtc:
|
||||||
|
ret = libcalamares.utils.target_env_call(hwclock_isa)
|
||||||
shutil.copy2("/etc/adjtime", "{!s}/etc/".format(root_mount_point))
|
if ret != 0:
|
||||||
|
is_broken_isa = True
|
||||||
|
libcalamares.utils.debug("Hwclock returned error code {}".format(ret))
|
||||||
|
libcalamares.utils.debug(" .. ISA bus method failed.")
|
||||||
|
else:
|
||||||
|
libcalamares.utils.debug("Hwclock set using ISA bus methode.")
|
||||||
|
if is_broken_rtc and is_broken_isa:
|
||||||
|
libcalamares.utils.debug("BIOS or Kernel BUG: Setting hwclock failed.")
|
||||||
|
|
||||||
return None
|
return None
|
||||||
|
@@ -1,5 +1,4 @@
|
|||||||
find_package(ECM 5.10.0 REQUIRED NO_MODULE)
|
find_package(ECM ${ECM_VERSION} REQUIRED NO_MODULE)
|
||||||
set(CMAKE_MODULE_PATH ${ECM_MODULE_PATH} ${CMAKE_MODULE_PATH})
|
|
||||||
|
|
||||||
include(KDEInstallDirs)
|
include(KDEInstallDirs)
|
||||||
include(GenerateExportHeader)
|
include(GenerateExportHeader)
|
||||||
|
@@ -1,9 +1,26 @@
|
|||||||
|
find_package(ECM ${ECM_VERSION} NO_MODULE)
|
||||||
|
if( ECM_FOUND AND BUILD_TESTING )
|
||||||
|
include( ECMAddTests )
|
||||||
|
find_package( Qt5 COMPONENTS Core Test REQUIRED )
|
||||||
|
endif()
|
||||||
|
|
||||||
include_directories( ${PROJECT_BINARY_DIR}/src/libcalamaresui )
|
include_directories( ${PROJECT_BINARY_DIR}/src/libcalamaresui )
|
||||||
|
|
||||||
|
set( geoip_src GeoIP.cpp GeoIPFreeGeoIP.cpp )
|
||||||
|
set( geoip_libs )
|
||||||
|
|
||||||
|
find_package(Qt5 COMPONENTS Xml)
|
||||||
|
if( Qt5Xml_FOUND )
|
||||||
|
list( APPEND geoip_src GeoIPXML.cpp )
|
||||||
|
list( APPEND geoip_libs Qt5::Xml )
|
||||||
|
add_definitions( -DHAVE_XML )
|
||||||
|
endif()
|
||||||
|
|
||||||
calamares_add_plugin( locale
|
calamares_add_plugin( locale
|
||||||
TYPE viewmodule
|
TYPE viewmodule
|
||||||
EXPORT_MACRO PLUGINDLLEXPORT_PRO
|
EXPORT_MACRO PLUGINDLLEXPORT_PRO
|
||||||
SOURCES
|
SOURCES
|
||||||
|
${geoip_src}
|
||||||
LCLocaleDialog.cpp
|
LCLocaleDialog.cpp
|
||||||
LocaleConfiguration.cpp
|
LocaleConfiguration.cpp
|
||||||
LocalePage.cpp
|
LocalePage.cpp
|
||||||
@@ -17,6 +34,28 @@ calamares_add_plugin( locale
|
|||||||
LINK_PRIVATE_LIBRARIES
|
LINK_PRIVATE_LIBRARIES
|
||||||
calamaresui
|
calamaresui
|
||||||
Qt5::Network
|
Qt5::Network
|
||||||
|
${geoip_libs}
|
||||||
${YAMLCPP_LIBRARY}
|
${YAMLCPP_LIBRARY}
|
||||||
SHARED_LIB
|
SHARED_LIB
|
||||||
)
|
)
|
||||||
|
|
||||||
|
if( ECM_FOUND AND BUILD_TESTING )
|
||||||
|
ecm_add_test(
|
||||||
|
GeoIPTests.cpp
|
||||||
|
${geoip_src}
|
||||||
|
TEST_NAME
|
||||||
|
geoiptest
|
||||||
|
LINK_LIBRARIES
|
||||||
|
calamaresui
|
||||||
|
Qt5::Network
|
||||||
|
Qt5::Test
|
||||||
|
${geoip_libs}
|
||||||
|
${YAMLCPP_LIBRARY}
|
||||||
|
)
|
||||||
|
set_target_properties( geoiptest PROPERTIES AUTOMOC TRUE )
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if( BUILD_TESTING )
|
||||||
|
add_executable( test_geoip test_geoip.cpp ${geoip_src} )
|
||||||
|
target_link_libraries( test_geoip calamaresui Qt5::Network ${geoip_libs} ${YAMLCPP_LIBRARY} )
|
||||||
|
endif()
|
||||||
|
40
src/modules/locale/GeoIP.cpp
Normal file
40
src/modules/locale/GeoIP.cpp
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2018, Adriaan de Groot <groot@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 "GeoIP.h"
|
||||||
|
|
||||||
|
#include "utils/Logger.h"
|
||||||
|
|
||||||
|
GeoIP::~GeoIP()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
GeoIP::RegionZonePair
|
||||||
|
GeoIP::splitTZString( const QString& timezoneString )
|
||||||
|
{
|
||||||
|
QStringList tzParts = timezoneString.split( '/', QString::SkipEmptyParts );
|
||||||
|
if ( tzParts.size() >= 2 )
|
||||||
|
{
|
||||||
|
cDebug() << "GeoIP reporting" << timezoneString;
|
||||||
|
QString region = tzParts.takeFirst();
|
||||||
|
QString zone = tzParts.join( '/' );
|
||||||
|
return qMakePair( region, zone );
|
||||||
|
}
|
||||||
|
|
||||||
|
return qMakePair( QString(), QString() );
|
||||||
|
}
|
56
src/modules/locale/GeoIP.h
Normal file
56
src/modules/locale/GeoIP.h
Normal file
@@ -0,0 +1,56 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2018, Adriaan de Groot <groot@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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GEOIP_H
|
||||||
|
#define GEOIP_H
|
||||||
|
|
||||||
|
#include <QPair>
|
||||||
|
#include <QString>
|
||||||
|
#include <QUrl>
|
||||||
|
|
||||||
|
class QByteArray;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Interface for GeoIP retrievers.
|
||||||
|
*
|
||||||
|
* A GeoIP retriever takes a configured URL (from the config file)
|
||||||
|
* and can handle the data returned from its interpretation of that
|
||||||
|
* configured URL, returning a region and zone.
|
||||||
|
*/
|
||||||
|
struct GeoIP
|
||||||
|
{
|
||||||
|
using RegionZonePair = QPair<QString, QString>;
|
||||||
|
|
||||||
|
virtual ~GeoIP();
|
||||||
|
|
||||||
|
/** @brief Handle a (successful) request by interpreting the data.
|
||||||
|
*
|
||||||
|
* Should return a ( <zone>, <region> ) pair, e.g.
|
||||||
|
* ( "Europe", "Amsterdam" ). This is called **only** if the
|
||||||
|
* request to the fullUrl was successful; the handler
|
||||||
|
* is free to read as much, or as little, data as it
|
||||||
|
* likes. On error, returns a RegionZonePair with empty
|
||||||
|
* strings (e.g. ( "", "" ) ).
|
||||||
|
*/
|
||||||
|
virtual RegionZonePair processReply( const QByteArray& ) = 0;
|
||||||
|
|
||||||
|
/** @brief Splits a region/zone string into a pair. */
|
||||||
|
static RegionZonePair splitTZString( const QString& s );
|
||||||
|
} ;
|
||||||
|
|
||||||
|
#endif
|
55
src/modules/locale/GeoIPFreeGeoIP.cpp
Normal file
55
src/modules/locale/GeoIPFreeGeoIP.cpp
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2014-2016, Teo Mrnjavac <teo@kde.org>
|
||||||
|
* Copyright 2018, Adriaan de Groot <groot@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 "GeoIPFreeGeoIP.h"
|
||||||
|
|
||||||
|
#include "utils/Logger.h"
|
||||||
|
#include "utils/YamlUtils.h"
|
||||||
|
|
||||||
|
#include <QByteArray>
|
||||||
|
|
||||||
|
#include <yaml-cpp/yaml.h>
|
||||||
|
|
||||||
|
GeoIP::RegionZonePair
|
||||||
|
FreeGeoIP::processReply( const QByteArray& data )
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
YAML::Node doc = YAML::Load( data );
|
||||||
|
|
||||||
|
QVariant var = CalamaresUtils::yamlToVariant( doc );
|
||||||
|
if ( !var.isNull() &&
|
||||||
|
var.isValid() &&
|
||||||
|
var.type() == QVariant::Map )
|
||||||
|
{
|
||||||
|
QVariantMap map = var.toMap();
|
||||||
|
if ( map.contains( "time_zone" ) &&
|
||||||
|
!map.value( "time_zone" ).toString().isEmpty() )
|
||||||
|
{
|
||||||
|
return splitTZString( map.value( "time_zone" ).toString() );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch ( YAML::Exception& e )
|
||||||
|
{
|
||||||
|
CalamaresUtils::explainYamlException( e, data, "GeoIP data");
|
||||||
|
}
|
||||||
|
|
||||||
|
return qMakePair( QString(), QString() );
|
||||||
|
}
|
37
src/modules/locale/GeoIPFreeGeoIP.h
Normal file
37
src/modules/locale/GeoIPFreeGeoIP.h
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2018, Adriaan de Groot <groot@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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GEOIPFREEGEOIP_H
|
||||||
|
#define GEOIPFREEGEOIP_H
|
||||||
|
|
||||||
|
#include "GeoIP.h"
|
||||||
|
|
||||||
|
/** @brief GeoIP lookup via freegeoip.com
|
||||||
|
*
|
||||||
|
* This is the original implementation of GeoIP lookup,
|
||||||
|
* using the FreeGeoIP service, or similar which returns
|
||||||
|
* data in the same format.
|
||||||
|
*
|
||||||
|
* The data is assumed to be in JSON format with a time_zone attribute.
|
||||||
|
*/
|
||||||
|
struct FreeGeoIP : public GeoIP
|
||||||
|
{
|
||||||
|
virtual RegionZonePair processReply( const QByteArray& );
|
||||||
|
} ;
|
||||||
|
|
||||||
|
#endif
|
140
src/modules/locale/GeoIPTests.cpp
Normal file
140
src/modules/locale/GeoIPTests.cpp
Normal file
@@ -0,0 +1,140 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2018, Adriaan de Groot <groot@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 "GeoIPTests.h"
|
||||||
|
|
||||||
|
#include "GeoIPFreeGeoIP.h"
|
||||||
|
#ifdef HAVE_XML
|
||||||
|
#include "GeoIPXML.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <QtTest/QtTest>
|
||||||
|
|
||||||
|
QTEST_GUILESS_MAIN( GeoIPTests )
|
||||||
|
|
||||||
|
GeoIPTests::GeoIPTests()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
GeoIPTests::~GeoIPTests()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
GeoIPTests::initTestCase()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
GeoIPTests::testJSON()
|
||||||
|
{
|
||||||
|
static const char data[] =
|
||||||
|
"{\"time_zone\":\"Europe/Amsterdam\"}";
|
||||||
|
|
||||||
|
FreeGeoIP handler;
|
||||||
|
auto tz = handler.processReply( data );
|
||||||
|
|
||||||
|
QCOMPARE( tz.first, QLatin1String( "Europe" ) );
|
||||||
|
QCOMPARE( tz.second, QLatin1String( "Amsterdam" ) );
|
||||||
|
|
||||||
|
// JSON is quite tolerant
|
||||||
|
tz = handler.processReply( "time_zone: \"Europe/Brussels\"" );
|
||||||
|
QCOMPARE( tz.second, QLatin1String( "Brussels" ) );
|
||||||
|
|
||||||
|
tz = handler.processReply( "time_zone: America/New_York\n" );
|
||||||
|
QCOMPARE( tz.first, QLatin1String( "America" ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
GeoIPTests::testJSONbad()
|
||||||
|
{
|
||||||
|
static const char data[] = "time_zone: 1";
|
||||||
|
|
||||||
|
FreeGeoIP handler;
|
||||||
|
auto tz = handler.processReply( data );
|
||||||
|
|
||||||
|
tz = handler.processReply( data );
|
||||||
|
QCOMPARE( tz.first, QString() );
|
||||||
|
|
||||||
|
tz = handler.processReply( "" );
|
||||||
|
QCOMPARE( tz.first, QString() );
|
||||||
|
|
||||||
|
tz = handler.processReply( "<html><body>404 Forbidden</body></html>" );
|
||||||
|
QCOMPARE( tz.first, QString() );
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void
|
||||||
|
GeoIPTests::testXML()
|
||||||
|
{
|
||||||
|
static const char data[] =
|
||||||
|
R"(<Response>
|
||||||
|
<Ip>85.150.1.1</Ip>
|
||||||
|
<Status>OK</Status>
|
||||||
|
<CountryCode>NL</CountryCode>
|
||||||
|
<CountryCode3>NLD</CountryCode3>
|
||||||
|
<CountryName>Netherlands</CountryName>
|
||||||
|
<RegionCode>None</RegionCode>
|
||||||
|
<RegionName>None</RegionName>
|
||||||
|
<City>None</City>
|
||||||
|
<ZipPostalCode/>
|
||||||
|
<Latitude>50.0</Latitude>
|
||||||
|
<Longitude>4.0</Longitude>
|
||||||
|
<AreaCode>0</AreaCode>
|
||||||
|
<TimeZone>Europe/Amsterdam</TimeZone>
|
||||||
|
</Response>)";
|
||||||
|
|
||||||
|
#ifdef HAVE_XML
|
||||||
|
XMLGeoIP handler;
|
||||||
|
auto tz = handler.processReply( data );
|
||||||
|
|
||||||
|
QCOMPARE( tz.first, QLatin1String( "Europe" ) );
|
||||||
|
QCOMPARE( tz.second, QLatin1String( "Amsterdam" ) );
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
GeoIPTests::testXML2()
|
||||||
|
{
|
||||||
|
static const char data[] =
|
||||||
|
"<Response><TimeZone>America/North Dakota/Beulah</TimeZone></Response>";
|
||||||
|
|
||||||
|
#ifdef HAVE_XML
|
||||||
|
XMLGeoIP handler;
|
||||||
|
auto tz = handler.processReply( data );
|
||||||
|
|
||||||
|
QCOMPARE( tz.first, QLatin1String( "America" ) );
|
||||||
|
QCOMPARE( tz.second, QLatin1String( "North Dakota/Beulah" ) );
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
GeoIPTests::testXMLbad()
|
||||||
|
{
|
||||||
|
#ifdef HAVE_XML
|
||||||
|
XMLGeoIP handler;
|
||||||
|
auto tz = handler.processReply( "{time_zone: \"Europe/Paris\"}" );
|
||||||
|
QCOMPARE( tz.first, QString() );
|
||||||
|
|
||||||
|
tz = handler.processReply( "<Response></Response>" );
|
||||||
|
QCOMPARE( tz.first, QString() );
|
||||||
|
|
||||||
|
tz = handler.processReply( "fnord<html/>" );
|
||||||
|
QCOMPARE( tz.first, QString() );
|
||||||
|
#endif
|
||||||
|
}
|
40
src/modules/locale/GeoIPTests.h
Normal file
40
src/modules/locale/GeoIPTests.h
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2018, Adriaan de Groot <groot@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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GEOIPTESTS_H
|
||||||
|
#define GEOIPTESTS_H
|
||||||
|
|
||||||
|
#include <QObject>
|
||||||
|
|
||||||
|
class GeoIPTests : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
public:
|
||||||
|
GeoIPTests();
|
||||||
|
~GeoIPTests() override;
|
||||||
|
|
||||||
|
private Q_SLOTS:
|
||||||
|
void initTestCase();
|
||||||
|
void testJSON();
|
||||||
|
void testJSONbad();
|
||||||
|
void testXML();
|
||||||
|
void testXML2();
|
||||||
|
void testXMLbad();
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
50
src/modules/locale/GeoIPXML.cpp
Normal file
50
src/modules/locale/GeoIPXML.cpp
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2018, Adriaan de Groot <groot@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 "GeoIPXML.h"
|
||||||
|
|
||||||
|
#include "utils/Logger.h"
|
||||||
|
|
||||||
|
#include <QNetworkReply>
|
||||||
|
#include <QtXml/QDomDocument>
|
||||||
|
|
||||||
|
GeoIP::RegionZonePair
|
||||||
|
XMLGeoIP::processReply( const QByteArray& data )
|
||||||
|
{
|
||||||
|
QString domError;
|
||||||
|
int errorLine, errorColumn;
|
||||||
|
|
||||||
|
QDomDocument doc;
|
||||||
|
if ( doc.setContent( data, false, &domError, &errorLine, &errorColumn ) )
|
||||||
|
{
|
||||||
|
const auto tzElements = doc.elementsByTagName( "TimeZone" );
|
||||||
|
cDebug() << "GeoIP found" << tzElements.length() << "elements";
|
||||||
|
for ( int it = 0; it < tzElements.length(); ++it )
|
||||||
|
{
|
||||||
|
auto e = tzElements.at(it).toElement();
|
||||||
|
return splitTZString( e.text() );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
cDebug() << "GeoIP XML data error:" << domError << "(line" << errorLine << errorColumn << ')';
|
||||||
|
}
|
||||||
|
|
||||||
|
return qMakePair( QString(), QString() );
|
||||||
|
|
||||||
|
}
|
36
src/modules/locale/GeoIPXML.h
Normal file
36
src/modules/locale/GeoIPXML.h
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2018, Adriaan de Groot <groot@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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GEOIPXML_H
|
||||||
|
#define GEOIPXML_H
|
||||||
|
|
||||||
|
#include "GeoIP.h"
|
||||||
|
|
||||||
|
/** @brief GeoIP lookup with XML data
|
||||||
|
*
|
||||||
|
* The data is assumed to be in XML format with a
|
||||||
|
* <Response><TimeZone></TimeZone></Response>
|
||||||
|
* element, which contains the text (string) for the region/zone. This
|
||||||
|
* format is expected by, e.g. the Ubiquity installer.
|
||||||
|
*/
|
||||||
|
struct XMLGeoIP : public GeoIP
|
||||||
|
{
|
||||||
|
virtual RegionZonePair processReply( const QByteArray& );
|
||||||
|
} ;
|
||||||
|
|
||||||
|
#endif
|
@@ -43,9 +43,11 @@ LocaleConfiguration::fromLanguageAndLocation( const QString& languageLocale,
|
|||||||
const QStringList& availableLocales,
|
const QStringList& availableLocales,
|
||||||
const QString& countryCode )
|
const QString& countryCode )
|
||||||
{
|
{
|
||||||
LocaleConfiguration lc = LocaleConfiguration();
|
LocaleConfiguration lc;
|
||||||
|
|
||||||
|
// Note that the documentation how this works is in packages.conf
|
||||||
QString language = languageLocale.split( '_' ).first();
|
QString language = languageLocale.split( '_' ).first();
|
||||||
lc.myLanguageLocaleBcp47 = QLocale(language).bcp47Name();
|
lc.myLanguageLocaleBcp47 = QLocale(language).bcp47Name().toLower();
|
||||||
|
|
||||||
QStringList linesForLanguage;
|
QStringList linesForLanguage;
|
||||||
for ( const QString &line : availableLocales )
|
for ( const QString &line : availableLocales )
|
||||||
@@ -288,7 +290,7 @@ LocaleConfiguration::isEmpty() const
|
|||||||
|
|
||||||
|
|
||||||
QMap< QString, QString >
|
QMap< QString, QString >
|
||||||
LocaleConfiguration::toMap()
|
LocaleConfiguration::toMap() const
|
||||||
{
|
{
|
||||||
QMap< QString, QString > map;
|
QMap< QString, QString > map;
|
||||||
|
|
||||||
@@ -324,3 +326,9 @@ LocaleConfiguration::toMap()
|
|||||||
|
|
||||||
return map;
|
return map;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QString
|
||||||
|
LocaleConfiguration::toBcp47() const
|
||||||
|
{
|
||||||
|
return myLanguageLocaleBcp47;
|
||||||
|
}
|
||||||
|
@@ -35,16 +35,21 @@ public:
|
|||||||
|
|
||||||
bool isEmpty() const;
|
bool isEmpty() const;
|
||||||
|
|
||||||
|
QMap< QString, QString > toMap() const;
|
||||||
|
// Note that the documentation how this works is in packages.conf
|
||||||
|
QString toBcp47() const;
|
||||||
|
|
||||||
// These become all uppercase in locale.conf, but we keep them lowercase here to
|
// These become all uppercase in locale.conf, but we keep them lowercase here to
|
||||||
// avoid confusion with locale.h.
|
// avoid confusion with locale.h.
|
||||||
QString lang, lc_numeric, lc_time, lc_monetary, lc_paper, lc_name, lc_address,
|
QString lang, lc_numeric, lc_time, lc_monetary, lc_paper, lc_name, lc_address,
|
||||||
lc_telephone, lc_measurement, lc_identification;
|
lc_telephone, lc_measurement, lc_identification;
|
||||||
QString myLanguageLocaleBcp47;
|
|
||||||
QMap< QString, QString > toMap();
|
|
||||||
|
|
||||||
// If the user has explicitly selected language (from the dialog)
|
// If the user has explicitly selected language (from the dialog)
|
||||||
// or numbers format, set these to avoid implicit changes to them.
|
// or numbers format, set these to avoid implicit changes to them.
|
||||||
bool explicit_lang, explicit_lc;
|
bool explicit_lang, explicit_lc;
|
||||||
|
|
||||||
|
private:
|
||||||
|
QString myLanguageLocaleBcp47;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // LOCALECONFIGURATION_H
|
#endif // LOCALECONFIGURATION_H
|
||||||
|
@@ -489,8 +489,10 @@ LocalePage::updateGlobalStorage()
|
|||||||
->insert( "locationRegion", location.region );
|
->insert( "locationRegion", location.region );
|
||||||
Calamares::JobQueue::instance()->globalStorage()
|
Calamares::JobQueue::instance()->globalStorage()
|
||||||
->insert( "locationZone", location.zone );
|
->insert( "locationZone", location.zone );
|
||||||
Calamares::JobQueue::instance()->globalStorage()
|
|
||||||
->insert( "locale", m_selectedLocaleConfiguration.myLanguageLocaleBcp47);
|
const QString bcp47 = m_selectedLocaleConfiguration.toBcp47();
|
||||||
|
Calamares::JobQueue::instance()->globalStorage()->insert( "locale", bcp47 );
|
||||||
|
cDebug() << "Updated locale globals, BCP47=" << bcp47;
|
||||||
|
|
||||||
// If we're in chroot mode (normal install mode), then we immediately set the
|
// If we're in chroot mode (normal install mode), then we immediately set the
|
||||||
// timezone on the live system.
|
// timezone on the live system.
|
||||||
|
@@ -1,6 +1,7 @@
|
|||||||
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
*
|
*
|
||||||
* Copyright 2014-2016, Teo Mrnjavac <teo@kde.org>
|
* Copyright 2014-2016, Teo Mrnjavac <teo@kde.org>
|
||||||
|
* Copyright 2018, Adriaan de Groot <groot.org>
|
||||||
*
|
*
|
||||||
* Calamares is free software: you can redistribute it and/or modify
|
* Calamares is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -18,11 +19,17 @@
|
|||||||
|
|
||||||
#include "LocaleViewStep.h"
|
#include "LocaleViewStep.h"
|
||||||
|
|
||||||
|
#include "GeoIP.h"
|
||||||
|
#include "GeoIPFreeGeoIP.h"
|
||||||
|
#ifdef HAVE_XML
|
||||||
|
#include "GeoIPXML.h"
|
||||||
|
#endif
|
||||||
|
#include "GlobalStorage.h"
|
||||||
|
#include "JobQueue.h"
|
||||||
#include "LocalePage.h"
|
#include "LocalePage.h"
|
||||||
|
|
||||||
#include "timezonewidget/localeglobal.h"
|
#include "timezonewidget/localeglobal.h"
|
||||||
#include "widgets/WaitingWidget.h"
|
#include "widgets/WaitingWidget.h"
|
||||||
#include "JobQueue.h"
|
|
||||||
#include "GlobalStorage.h"
|
|
||||||
|
|
||||||
#include "utils/CalamaresUtilsGui.h"
|
#include "utils/CalamaresUtilsGui.h"
|
||||||
#include "utils/Logger.h"
|
#include "utils/Logger.h"
|
||||||
@@ -110,54 +117,52 @@ LocaleViewStep::setUpPage()
|
|||||||
void
|
void
|
||||||
LocaleViewStep::fetchGeoIpTimezone()
|
LocaleViewStep::fetchGeoIpTimezone()
|
||||||
{
|
{
|
||||||
|
QString actualUrl( m_geoipUrl );
|
||||||
|
GeoIP *handler = nullptr;
|
||||||
|
|
||||||
|
if ( m_geoipStyle.isEmpty() || m_geoipStyle == "legacy" )
|
||||||
|
{
|
||||||
|
actualUrl.append( "/json/" );
|
||||||
|
handler = new FreeGeoIP;
|
||||||
|
}
|
||||||
|
else if ( m_geoipStyle == "json" )
|
||||||
|
{
|
||||||
|
handler = new FreeGeoIP;
|
||||||
|
}
|
||||||
|
#if defined(HAVE_XML)
|
||||||
|
else if ( m_geoipStyle == "xml" )
|
||||||
|
{
|
||||||
|
handler = new XMLGeoIP;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
else
|
||||||
|
{
|
||||||
|
cDebug() << "WARNING: GeoIP Style" << m_geoipStyle << "is not recognized.";
|
||||||
|
setUpPage();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
cDebug() << "Fetching GeoIP data from" << actualUrl;
|
||||||
|
|
||||||
QNetworkAccessManager *manager = new QNetworkAccessManager( this );
|
QNetworkAccessManager *manager = new QNetworkAccessManager( this );
|
||||||
connect( manager, &QNetworkAccessManager::finished,
|
connect( manager, &QNetworkAccessManager::finished,
|
||||||
[=]( QNetworkReply* reply )
|
[=]( QNetworkReply* reply )
|
||||||
{
|
{
|
||||||
if ( reply->error() == QNetworkReply::NoError )
|
if ( reply->error() == QNetworkReply::NoError )
|
||||||
{
|
{
|
||||||
QByteArray data = reply->readAll();
|
auto tz = handler->processReply( reply->readAll() );
|
||||||
|
if ( !tz.first.isEmpty() )
|
||||||
try
|
m_startingTimezone = tz;
|
||||||
{
|
else
|
||||||
YAML::Node doc = YAML::Load( data );
|
cDebug() << "WARNING: GeoIP lookup at" << reply->url() << "failed.";
|
||||||
|
|
||||||
QVariant var = CalamaresUtils::yamlToVariant( doc );
|
|
||||||
if ( !var.isNull() &&
|
|
||||||
var.isValid() &&
|
|
||||||
var.type() == QVariant::Map )
|
|
||||||
{
|
|
||||||
QVariantMap map = var.toMap();
|
|
||||||
if ( map.contains( "time_zone" ) &&
|
|
||||||
!map.value( "time_zone" ).toString().isEmpty() )
|
|
||||||
{
|
|
||||||
QString timezoneString = map.value( "time_zone" ).toString();
|
|
||||||
QStringList tzParts = timezoneString.split( '/', QString::SkipEmptyParts );
|
|
||||||
if ( tzParts.size() >= 2 )
|
|
||||||
{
|
|
||||||
cDebug() << "GeoIP reporting" << timezoneString;
|
|
||||||
QString region = tzParts.takeFirst();
|
|
||||||
QString zone = tzParts.join( '/' );
|
|
||||||
m_startingTimezone = qMakePair( region, zone );
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch ( YAML::Exception& e )
|
|
||||||
{
|
|
||||||
CalamaresUtils::explainYamlException( e, data, "GeoIP data");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
delete handler;
|
||||||
reply->deleteLater();
|
reply->deleteLater();
|
||||||
manager->deleteLater();
|
manager->deleteLater();
|
||||||
setUpPage();
|
setUpPage();
|
||||||
} );
|
} );
|
||||||
|
|
||||||
QNetworkRequest request;
|
QNetworkRequest request;
|
||||||
QString requestUrl = QString( "%1/json" )
|
request.setUrl( QUrl::fromUserInput( actualUrl ) );
|
||||||
.arg( QUrl::fromUserInput( m_geoipUrl ).toString() );
|
|
||||||
request.setUrl( QUrl( requestUrl ) );
|
|
||||||
request.setAttribute( QNetworkRequest::FollowRedirectsAttribute, true );
|
request.setAttribute( QNetworkRequest::FollowRedirectsAttribute, true );
|
||||||
manager->get( request );
|
manager->get( request );
|
||||||
}
|
}
|
||||||
@@ -293,4 +298,10 @@ LocaleViewStep::setConfigurationMap( const QVariantMap& configurationMap )
|
|||||||
{
|
{
|
||||||
m_geoipUrl = configurationMap.value( "geoipUrl" ).toString();
|
m_geoipUrl = configurationMap.value( "geoipUrl" ).toString();
|
||||||
}
|
}
|
||||||
|
if ( configurationMap.contains( "geoipStyle" ) &&
|
||||||
|
configurationMap.value( "geoipStyle" ).type() == QVariant::String &&
|
||||||
|
!configurationMap.value( "geoipStyle" ).toString().isEmpty() )
|
||||||
|
{
|
||||||
|
m_geoipStyle = configurationMap.value( "geoipStyle" ).toString();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@@ -76,6 +76,7 @@ private:
|
|||||||
QPair< QString, QString > m_startingTimezone;
|
QPair< QString, QString > m_startingTimezone;
|
||||||
QString m_localeGenPath;
|
QString m_localeGenPath;
|
||||||
QString m_geoipUrl;
|
QString m_geoipUrl;
|
||||||
|
QString m_geoipStyle;
|
||||||
|
|
||||||
QList< Calamares::job_ptr > m_jobs;
|
QList< Calamares::job_ptr > m_jobs;
|
||||||
};
|
};
|
||||||
|
@@ -1,7 +1,56 @@
|
|||||||
---
|
---
|
||||||
|
# The starting timezone (e.g. the pin-on-the-map) when entering
|
||||||
|
# the locale page can be set through keys *region* and *zone*.
|
||||||
|
# If either is not set, defaults to America/New_York.
|
||||||
|
#
|
||||||
region: "America"
|
region: "America"
|
||||||
zone: "New_York"
|
zone: "New_York"
|
||||||
|
|
||||||
# GeoIP settings. Leave commented out to disable GeoIP.
|
# Some distros come with a meaningfully commented and easy to parse
|
||||||
|
# `/etc/locale.gen`, and others ship a separate file
|
||||||
|
# `/usr/share/i18n/SUPPORTED` with a clean list of supported locales.
|
||||||
|
# We first try SUPPORTED, and if it doesn't exist, we fall back
|
||||||
|
# to parsing the lines from `locale.gen`. For distro's that ship
|
||||||
|
# the `locale.gen` file installed elsewhere, the key *localeGenPath*
|
||||||
|
# can be used to specify where it is. If not set, the default
|
||||||
|
# `/etc/locale.gen` is used.
|
||||||
|
#
|
||||||
#localeGenPath: "/etc/locale.gen"
|
#localeGenPath: "/etc/locale.gen"
|
||||||
|
|
||||||
|
# GeoIP settings. Leave commented out to disable GeoIP.
|
||||||
|
#
|
||||||
|
# An HTTP request is made to *geoipUrl* -- depending on the geoipStyle,
|
||||||
|
# the URL may be modified before use. The request should return
|
||||||
|
# valid data in a suitable format, depending on geoipStyle;
|
||||||
|
# generally this includes a string value with the timezone
|
||||||
|
# in <region>/<zone> format.
|
||||||
|
#
|
||||||
|
# Note that this example URL works, but the service is shutting
|
||||||
|
# down in June 2018.
|
||||||
|
#
|
||||||
|
# Suitable JSON data looks like
|
||||||
|
# ```
|
||||||
|
# {"time_zone":"America/New_York"}
|
||||||
|
# ```
|
||||||
|
# Suitable XML data looks like
|
||||||
|
# ```
|
||||||
|
# <Response><TimeZone>Europe/Brussels</TimeZone></Response>
|
||||||
|
# ```
|
||||||
|
#
|
||||||
#geoipUrl: "freegeoip.net"
|
#geoipUrl: "freegeoip.net"
|
||||||
|
|
||||||
|
# GeoIP style. Leave commented out for the "legacy" interpretation.
|
||||||
|
# This setting only makes sense if geoipUrl is set, enabliing geoIP.
|
||||||
|
#
|
||||||
|
# Possible values are:
|
||||||
|
# unset same as "legacy"
|
||||||
|
# blank same as "legacy"
|
||||||
|
# "legacy" appends "/json" to geoipUrl, above, and uses JSON format
|
||||||
|
# (which is what freegeoip.net provides there).
|
||||||
|
# "json" URL is not modified, uses JSON format.
|
||||||
|
# "xml" URL is not modified, uses XML format.
|
||||||
|
#
|
||||||
|
# The JSON format is provided by freegeoip.net, but that service is
|
||||||
|
# shutting down in June 2018. There are other providers with the same
|
||||||
|
# format. XML format is provided for Ubiquity.
|
||||||
|
#geoipStyle: "legacy"
|
||||||
|
73
src/modules/locale/test_geoip.cpp
Normal file
73
src/modules/locale/test_geoip.cpp
Normal file
@@ -0,0 +1,73 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2018, Adriaan de Groot <groot@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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is a test-application that does one GeoIP parse.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <iostream>
|
||||||
|
|
||||||
|
#include "GeoIPFreeGeoIP.h"
|
||||||
|
#ifdef HAVE_XML
|
||||||
|
#include "GeoIPXML.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
using std::cerr;
|
||||||
|
|
||||||
|
int main(int argc, char** argv)
|
||||||
|
{
|
||||||
|
if (argc != 2)
|
||||||
|
{
|
||||||
|
cerr << "Usage: curl url | test_geoip <format>\n";
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
GeoIP* handler = nullptr;
|
||||||
|
if ( QLatin1String( "json" ) == argv[1] )
|
||||||
|
handler = new FreeGeoIP;
|
||||||
|
#ifdef HAVE_XML
|
||||||
|
else if ( QLatin1String( "xml" ) == argv[1] )
|
||||||
|
handler = new XMLGeoIP;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if ( !handler )
|
||||||
|
{
|
||||||
|
cerr << "Unknown format '" << argv[1] << "'\n";
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
QByteArray ba;
|
||||||
|
while( !std::cin.eof() ) {
|
||||||
|
char arr[1024];
|
||||||
|
std::cin.read(arr,sizeof(arr));
|
||||||
|
int s = std::cin.gcount();
|
||||||
|
ba.append(arr, s);
|
||||||
|
}
|
||||||
|
|
||||||
|
auto tz = handler->processReply( ba );
|
||||||
|
if ( tz.first.isEmpty() )
|
||||||
|
{
|
||||||
|
std::cout << "No TimeZone determined from input.\n";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
std::cout << "TimeZone Region=" << tz.first.toLatin1().constData() << "\nTimeZone Zone=" << tz.second.toLatin1().constData() << '\n';
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
@@ -318,7 +318,10 @@ def subst_locale(plist):
|
|||||||
"""
|
"""
|
||||||
locale = libcalamares.globalstorage.value("locale")
|
locale = libcalamares.globalstorage.value("locale")
|
||||||
if not locale:
|
if not locale:
|
||||||
return plist
|
# It is possible to skip the locale-setting entirely.
|
||||||
|
# Then pretend it is "en", so that {LOCALE}-decorated
|
||||||
|
# package names are removed from the list.
|
||||||
|
locale = "en"
|
||||||
|
|
||||||
ret = []
|
ret = []
|
||||||
for packagedata in plist:
|
for packagedata in plist:
|
||||||
@@ -364,20 +367,20 @@ def run_operations(pkgman, entry):
|
|||||||
global group_packages, completed_packages, mode_packages
|
global group_packages, completed_packages, mode_packages
|
||||||
|
|
||||||
for key in entry.keys():
|
for key in entry.keys():
|
||||||
entry[key] = subst_locale(entry[key])
|
package_list = subst_locale(entry[key])
|
||||||
group_packages = len(entry[key])
|
group_packages = len(package_list)
|
||||||
if key == "install":
|
if key == "install":
|
||||||
_change_mode(INSTALL)
|
_change_mode(INSTALL)
|
||||||
if all([isinstance(x, str) for x in entry[key]]):
|
if all([isinstance(x, str) for x in package_list]):
|
||||||
pkgman.install(entry[key])
|
pkgman.install(package_list)
|
||||||
else:
|
else:
|
||||||
for package in entry[key]:
|
for package in package_list:
|
||||||
pkgman.install_package(package)
|
pkgman.install_package(package)
|
||||||
elif key == "try_install":
|
elif key == "try_install":
|
||||||
_change_mode(INSTALL)
|
_change_mode(INSTALL)
|
||||||
# we make a separate package manager call for each package so a
|
# we make a separate package manager call for each package so a
|
||||||
# single failing package won't stop all of them
|
# single failing package won't stop all of them
|
||||||
for package in entry[key]:
|
for package in package_list:
|
||||||
try:
|
try:
|
||||||
pkgman.install_package(package)
|
pkgman.install_package(package)
|
||||||
except subprocess.CalledProcessError:
|
except subprocess.CalledProcessError:
|
||||||
@@ -386,10 +389,10 @@ def run_operations(pkgman, entry):
|
|||||||
libcalamares.utils.debug(warn_text)
|
libcalamares.utils.debug(warn_text)
|
||||||
elif key == "remove":
|
elif key == "remove":
|
||||||
_change_mode(REMOVE)
|
_change_mode(REMOVE)
|
||||||
pkgman.remove(entry[key])
|
pkgman.remove(package_list)
|
||||||
elif key == "try_remove":
|
elif key == "try_remove":
|
||||||
_change_mode(REMOVE)
|
_change_mode(REMOVE)
|
||||||
for package in entry[key]:
|
for package in package_list:
|
||||||
try:
|
try:
|
||||||
pkgman.remove([package])
|
pkgman.remove([package])
|
||||||
except subprocess.CalledProcessError:
|
except subprocess.CalledProcessError:
|
||||||
@@ -398,9 +401,9 @@ def run_operations(pkgman, entry):
|
|||||||
libcalamares.utils.debug(warn_text)
|
libcalamares.utils.debug(warn_text)
|
||||||
elif key == "localInstall":
|
elif key == "localInstall":
|
||||||
_change_mode(INSTALL)
|
_change_mode(INSTALL)
|
||||||
pkgman.install(entry[key], from_local=True)
|
pkgman.install(package_list, from_local=True)
|
||||||
|
|
||||||
completed_packages += len(entry[key])
|
completed_packages += len(package_list)
|
||||||
libcalamares.job.setprogress(completed_packages * 1.0 / total_packages)
|
libcalamares.job.setprogress(completed_packages * 1.0 / total_packages)
|
||||||
libcalamares.utils.debug(pretty_name())
|
libcalamares.utils.debug(pretty_name())
|
||||||
|
|
||||||
@@ -444,7 +447,7 @@ def run():
|
|||||||
completed_packages = 0
|
completed_packages = 0
|
||||||
for op in operations:
|
for op in operations:
|
||||||
for packagelist in op.values():
|
for packagelist in op.values():
|
||||||
total_packages += len(packagelist)
|
total_packages += len(subst_locale(packagelist))
|
||||||
|
|
||||||
if not total_packages:
|
if not total_packages:
|
||||||
# Avoids potential divide-by-zero in progress reporting
|
# Avoids potential divide-by-zero in progress reporting
|
||||||
|
@@ -76,7 +76,7 @@ update_db: true
|
|||||||
# pre-script: touch /tmp/installing-vi
|
# pre-script: touch /tmp/installing-vi
|
||||||
# post-script: rm -f /tmp/installing-vi
|
# post-script: rm -f /tmp/installing-vi
|
||||||
#
|
#
|
||||||
# The pre- and post-scripts are optional, but not both optional: using
|
# The pre- and post-scripts are optional, but you cannot leave both out: using
|
||||||
# "package: vi" with neither script option will trick Calamares into
|
# "package: vi" with neither script option will trick Calamares into
|
||||||
# trying to install a package named "package: vi", which is unlikely to work.
|
# trying to install a package named "package: vi", which is unlikely to work.
|
||||||
#
|
#
|
||||||
@@ -84,11 +84,16 @@ update_db: true
|
|||||||
# packages for software based on the selected system locale. By including
|
# packages for software based on the selected system locale. By including
|
||||||
# the string LOCALE in the package name, the following happens:
|
# the string LOCALE in the package name, the following happens:
|
||||||
#
|
#
|
||||||
# - if the system locale is English (generally US English; en_GB is a valid
|
# - if the system locale is English (any variety), then the package is not
|
||||||
# localization), then the package is not installed at all,
|
# installed at all,
|
||||||
# - otherwise $LOCALE or ${LOCALE} is replaced by the Bcp47 name of the selected
|
# - otherwise $LOCALE or ${LOCALE} is replaced by the **lower-cased** BCP47
|
||||||
# system locale, e.g. nl_BE. Note that just plain LOCALE will not be replaced,
|
# name of the **language** part of the selected system locale (not the
|
||||||
# so foo-LOCALE will be unchanged, while foo-$LOCALE will be changed.
|
# country/region/dialect part), e.g. selecting *nl_BE* will use *nl*
|
||||||
|
# here.
|
||||||
|
#
|
||||||
|
# Take care that just plain LOCALE will not be replaced, so foo-LOCALE will
|
||||||
|
# be left unchanged, while foo-$LOCALE will be changed. However, foo-LOCALE
|
||||||
|
# **will** be removed from the list of packages, if English is selected.
|
||||||
#
|
#
|
||||||
# The following installs localizations for vi, if they are relevant; if
|
# The following installs localizations for vi, if they are relevant; if
|
||||||
# there is no localization, installation continues normally.
|
# there is no localization, installation continues normally.
|
||||||
@@ -127,6 +132,7 @@ update_db: true
|
|||||||
operations:
|
operations:
|
||||||
- install:
|
- install:
|
||||||
- vi
|
- vi
|
||||||
|
- vi-${LOCALE}
|
||||||
- wget
|
- wget
|
||||||
- binutils
|
- binutils
|
||||||
- remove:
|
- remove:
|
||||||
|
@@ -1,5 +1,4 @@
|
|||||||
find_package(ECM 5.10.0 REQUIRED NO_MODULE)
|
find_package(ECM ${ECM_VERSION} REQUIRED NO_MODULE)
|
||||||
set(CMAKE_MODULE_PATH ${ECM_MODULE_PATH} ${CMAKE_MODULE_PATH})
|
|
||||||
|
|
||||||
include(KDEInstallDirs)
|
include(KDEInstallDirs)
|
||||||
include(GenerateExportHeader)
|
include(GenerateExportHeader)
|
||||||
|
55
src/modules/plasmalnf/CMakeLists.txt
Normal file
55
src/modules/plasmalnf/CMakeLists.txt
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
find_package(ECM ${ECM_VERSION} REQUIRED NO_MODULE)
|
||||||
|
|
||||||
|
# Requires a sufficiently recent Plasma framework, but also
|
||||||
|
# needs a runtime support component (which we don't test for).
|
||||||
|
set( lnf_ver 5.41 )
|
||||||
|
|
||||||
|
find_package( KF5Config ${lnf_ver} )
|
||||||
|
find_package( KF5Plasma ${lnf_ver} )
|
||||||
|
find_package( KF5Package ${lnf_ver} )
|
||||||
|
set_package_properties(
|
||||||
|
KF5Config PROPERTIES
|
||||||
|
PURPOSE "For finding default Plasma Look-and-Feel"
|
||||||
|
)
|
||||||
|
set_package_properties(
|
||||||
|
KF5Plasma PROPERTIES
|
||||||
|
PURPOSE "For Plasma Look-and-Feel selection"
|
||||||
|
)
|
||||||
|
set_package_properties(
|
||||||
|
KF5Package PROPERTIES
|
||||||
|
PURPOSE "For Plasma Look-and-Feel selection"
|
||||||
|
)
|
||||||
|
|
||||||
|
if ( KF5Plasma_FOUND AND KF5Package_FOUND )
|
||||||
|
if ( KF5Config_FOUND )
|
||||||
|
set( option_kf5 Config )
|
||||||
|
set( option_defs WITH_KCONFIG )
|
||||||
|
# set( option_libs KF5::Config ) # Not needed anyway
|
||||||
|
endif()
|
||||||
|
|
||||||
|
find_package( KF5 ${lnf_ver} REQUIRED CoreAddons Plasma Package ${option_kf5} )
|
||||||
|
|
||||||
|
calamares_add_plugin( plasmalnf
|
||||||
|
TYPE viewmodule
|
||||||
|
EXPORT_MACRO PLUGINDLLEXPORT_PRO
|
||||||
|
COMPILE_DEFINITIONS
|
||||||
|
${option_defs}
|
||||||
|
SOURCES
|
||||||
|
PlasmaLnfViewStep.cpp
|
||||||
|
PlasmaLnfPage.cpp
|
||||||
|
PlasmaLnfJob.cpp
|
||||||
|
ThemeWidget.cpp
|
||||||
|
RESOURCES
|
||||||
|
page_plasmalnf.qrc
|
||||||
|
UI
|
||||||
|
page_plasmalnf.ui
|
||||||
|
LINK_PRIVATE_LIBRARIES
|
||||||
|
calamaresui
|
||||||
|
${option_libs}
|
||||||
|
KF5::Package
|
||||||
|
KF5::Plasma
|
||||||
|
SHARED_LIB
|
||||||
|
)
|
||||||
|
else()
|
||||||
|
calamares_skip_module( "plasmalnf (missing requirements)" )
|
||||||
|
endif()
|
78
src/modules/plasmalnf/PlasmaLnfJob.cpp
Normal file
78
src/modules/plasmalnf/PlasmaLnfJob.cpp
Normal file
@@ -0,0 +1,78 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2017, Adriaan de Groot <groot@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 "PlasmaLnfJob.h"
|
||||||
|
|
||||||
|
#include "GlobalStorage.h"
|
||||||
|
#include "JobQueue.h"
|
||||||
|
#include "utils/CalamaresUtilsSystem.h"
|
||||||
|
#include "utils/Logger.h"
|
||||||
|
|
||||||
|
PlasmaLnfJob::PlasmaLnfJob( const QString& lnfPath, const QString& id )
|
||||||
|
: m_lnfPath( lnfPath )
|
||||||
|
, m_id( id )
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
PlasmaLnfJob::~PlasmaLnfJob()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
QString
|
||||||
|
PlasmaLnfJob::prettyName() const
|
||||||
|
{
|
||||||
|
return tr( "Plasma Look-and-Feel Job" );
|
||||||
|
}
|
||||||
|
|
||||||
|
QString
|
||||||
|
PlasmaLnfJob::prettyDescription() const
|
||||||
|
{
|
||||||
|
return prettyName();
|
||||||
|
}
|
||||||
|
|
||||||
|
QString PlasmaLnfJob::prettyStatusMessage() const
|
||||||
|
{
|
||||||
|
return prettyName();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Calamares::JobResult
|
||||||
|
PlasmaLnfJob::exec()
|
||||||
|
{
|
||||||
|
cDebug() << "Plasma Look-and-Feel Job";
|
||||||
|
|
||||||
|
auto system = CalamaresUtils::System::instance();
|
||||||
|
Calamares::GlobalStorage* gs = Calamares::JobQueue::instance()->globalStorage();
|
||||||
|
|
||||||
|
QStringList command(
|
||||||
|
{
|
||||||
|
"sudo", "-E", "-H", "-u", gs->value( "username" ).toString(),
|
||||||
|
m_lnfPath, "-platform", "minimal", "--resetLayout", "--apply", m_id
|
||||||
|
} );
|
||||||
|
|
||||||
|
int r = system->targetEnvCall( command );
|
||||||
|
if ( r )
|
||||||
|
return Calamares::JobResult::error(
|
||||||
|
tr( "Could not select KDE Plasma Look-and-Feel package" ),
|
||||||
|
tr( "Could not select KDE Plasma Look-and-Feel package" ) );
|
||||||
|
|
||||||
|
return Calamares::JobResult::ok();
|
||||||
|
}
|
||||||
|
|
46
src/modules/plasmalnf/PlasmaLnfJob.h
Normal file
46
src/modules/plasmalnf/PlasmaLnfJob.h
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2017, Adriaan de Groot <groot@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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef PLASMALNFJOB_H
|
||||||
|
#define PLASMALNFJOB_H
|
||||||
|
|
||||||
|
#include <QObject>
|
||||||
|
#include <QVariantMap>
|
||||||
|
|
||||||
|
#include <Job.h>
|
||||||
|
|
||||||
|
class PlasmaLnfJob : public Calamares::Job
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit PlasmaLnfJob( const QString& lnfPath, const QString& id );
|
||||||
|
virtual ~PlasmaLnfJob() override;
|
||||||
|
|
||||||
|
QString prettyName() const override;
|
||||||
|
QString prettyDescription() const override;
|
||||||
|
QString prettyStatusMessage() const override;
|
||||||
|
|
||||||
|
Calamares::JobResult exec() override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
QString m_lnfPath;
|
||||||
|
QString m_id;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // PLASMALNFJOB_H
|
196
src/modules/plasmalnf/PlasmaLnfPage.cpp
Normal file
196
src/modules/plasmalnf/PlasmaLnfPage.cpp
Normal file
@@ -0,0 +1,196 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2017, Adriaan de Groot <groot@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 "PlasmaLnfPage.h"
|
||||||
|
|
||||||
|
#include "ui_page_plasmalnf.h"
|
||||||
|
|
||||||
|
#include "utils/Logger.h"
|
||||||
|
#include "utils/Retranslator.h"
|
||||||
|
|
||||||
|
#include <QAbstractButton>
|
||||||
|
|
||||||
|
#include <KPackage/Package>
|
||||||
|
#include <KPackage/PackageLoader>
|
||||||
|
|
||||||
|
ThemeInfo::ThemeInfo( const KPluginMetaData& data )
|
||||||
|
: id( data.pluginId() )
|
||||||
|
, name( data.name() )
|
||||||
|
, description( data.description() )
|
||||||
|
, widget( nullptr )
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
static ThemeInfoList plasma_themes()
|
||||||
|
{
|
||||||
|
ThemeInfoList packages;
|
||||||
|
|
||||||
|
QList<KPluginMetaData> pkgs = KPackage::PackageLoader::self()->listPackages( "Plasma/LookAndFeel" );
|
||||||
|
|
||||||
|
for ( const KPluginMetaData& data : pkgs )
|
||||||
|
{
|
||||||
|
if ( data.isValid() && !data.isHidden() && !data.name().isEmpty() )
|
||||||
|
{
|
||||||
|
packages << ThemeInfo{ data };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return packages;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
PlasmaLnfPage::PlasmaLnfPage( QWidget* parent )
|
||||||
|
: QWidget( parent )
|
||||||
|
, ui( new Ui::PlasmaLnfPage )
|
||||||
|
, m_showAll( false )
|
||||||
|
, m_buttonGroup( nullptr )
|
||||||
|
{
|
||||||
|
ui->setupUi( this );
|
||||||
|
CALAMARES_RETRANSLATE(
|
||||||
|
{
|
||||||
|
ui->retranslateUi( this );
|
||||||
|
ui->generalExplanation->setText( tr(
|
||||||
|
"Please choose a look-and-feel for the KDE Plasma Desktop. "
|
||||||
|
"You can also skip this step and configure the look-and-feel "
|
||||||
|
"once the system is installed. Clicking on a look-and-feel "
|
||||||
|
"selection will give you a live preview of that look-and-feel.") );
|
||||||
|
updateThemeNames();
|
||||||
|
fillUi();
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
PlasmaLnfPage::setLnfPath( const QString& path )
|
||||||
|
{
|
||||||
|
m_lnfPath = path;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
PlasmaLnfPage::setEnabledThemes(const ThemeInfoList& themes, bool showAll )
|
||||||
|
{
|
||||||
|
m_enabledThemes = themes;
|
||||||
|
|
||||||
|
if ( showAll )
|
||||||
|
{
|
||||||
|
auto plasmaThemes = plasma_themes();
|
||||||
|
for ( auto& installed_theme : plasmaThemes )
|
||||||
|
if ( !m_enabledThemes.findById( installed_theme.id ) )
|
||||||
|
m_enabledThemes.append( installed_theme );
|
||||||
|
}
|
||||||
|
|
||||||
|
updateThemeNames();
|
||||||
|
winnowThemes();
|
||||||
|
fillUi();
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
PlasmaLnfPage::setEnabledThemesAll()
|
||||||
|
{
|
||||||
|
// Don't need to set showAll=true, because we're already passing in
|
||||||
|
// the complete list of installed themes.
|
||||||
|
setEnabledThemes( plasma_themes(), false );
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
PlasmaLnfPage::setPreselect( const QString& id )
|
||||||
|
{
|
||||||
|
m_preselect = id;
|
||||||
|
if ( !m_enabledThemes.isEmpty() )
|
||||||
|
fillUi();
|
||||||
|
}
|
||||||
|
|
||||||
|
void PlasmaLnfPage::updateThemeNames()
|
||||||
|
{
|
||||||
|
auto plasmaThemes = plasma_themes();
|
||||||
|
for ( auto& enabled_theme : m_enabledThemes )
|
||||||
|
{
|
||||||
|
ThemeInfo* t = plasmaThemes.findById( enabled_theme.id );
|
||||||
|
if ( t != nullptr )
|
||||||
|
{
|
||||||
|
enabled_theme.name = t->name;
|
||||||
|
enabled_theme.description = t->description;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void PlasmaLnfPage::winnowThemes()
|
||||||
|
{
|
||||||
|
auto plasmaThemes = plasma_themes();
|
||||||
|
bool winnowed = true;
|
||||||
|
int winnow_index = 0;
|
||||||
|
while ( winnowed )
|
||||||
|
{
|
||||||
|
winnowed = false;
|
||||||
|
winnow_index = 0;
|
||||||
|
|
||||||
|
for ( auto& enabled_theme : m_enabledThemes )
|
||||||
|
{
|
||||||
|
ThemeInfo* t = plasmaThemes.findById( enabled_theme.id );
|
||||||
|
if ( t == nullptr )
|
||||||
|
{
|
||||||
|
cDebug() << "Removing" << enabled_theme.id;
|
||||||
|
winnowed = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
++winnow_index;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( winnowed )
|
||||||
|
{
|
||||||
|
m_enabledThemes.removeAt( winnow_index );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void PlasmaLnfPage::fillUi()
|
||||||
|
{
|
||||||
|
if ( m_enabledThemes.isEmpty() )
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( !m_buttonGroup )
|
||||||
|
{
|
||||||
|
m_buttonGroup = new QButtonGroup( this );
|
||||||
|
m_buttonGroup->setExclusive( true );
|
||||||
|
}
|
||||||
|
|
||||||
|
int c = 1; // After the general explanation
|
||||||
|
for ( auto& theme : m_enabledThemes )
|
||||||
|
{
|
||||||
|
if ( !theme.widget )
|
||||||
|
{
|
||||||
|
ThemeWidget* w = new ThemeWidget( theme );
|
||||||
|
m_buttonGroup->addButton( w->button() );
|
||||||
|
ui->verticalLayout->insertWidget( c, w );
|
||||||
|
connect( w, &ThemeWidget::themeSelected, this, &PlasmaLnfPage::plasmaThemeSelected);
|
||||||
|
theme.widget = w;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
theme.widget->updateThemeName( theme );
|
||||||
|
}
|
||||||
|
if ( theme.id == m_preselect )
|
||||||
|
{
|
||||||
|
const QSignalBlocker b( theme.widget->button() );
|
||||||
|
theme.widget->button()->setChecked( true );
|
||||||
|
}
|
||||||
|
++c;
|
||||||
|
}
|
||||||
|
}
|
82
src/modules/plasmalnf/PlasmaLnfPage.h
Normal file
82
src/modules/plasmalnf/PlasmaLnfPage.h
Normal file
@@ -0,0 +1,82 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2017, Adriaan de Groot <groot@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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef PLASMALNFPAGE_H
|
||||||
|
#define PLASMALNFPAGE_H
|
||||||
|
|
||||||
|
#include <QButtonGroup>
|
||||||
|
#include <QList>
|
||||||
|
#include <QString>
|
||||||
|
#include <QStringList>
|
||||||
|
#include <QWidget>
|
||||||
|
|
||||||
|
#include "ThemeInfo.h"
|
||||||
|
#include "ThemeWidget.h"
|
||||||
|
|
||||||
|
namespace Ui
|
||||||
|
{
|
||||||
|
class PlasmaLnfPage;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @brief Page for selecting a Plasma Look-and-Feel theme.
|
||||||
|
*
|
||||||
|
* You must call setEnabledThemes -- either overload -- once
|
||||||
|
* to get the selection widgets. Note that calling that with
|
||||||
|
* an empty list will result in zero (0) selectable themes.
|
||||||
|
*/
|
||||||
|
class PlasmaLnfPage : public QWidget
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
public:
|
||||||
|
explicit PlasmaLnfPage( QWidget* parent = nullptr );
|
||||||
|
|
||||||
|
void setLnfPath( const QString& path );
|
||||||
|
/** @brief enable only the listed themes.
|
||||||
|
*
|
||||||
|
* Shows the listed @p themes with full information (e.g. screenshot).
|
||||||
|
* If @p showAll is true, then also show all installed themes
|
||||||
|
* not explicitly listed (without a screenshot).
|
||||||
|
*/
|
||||||
|
void setEnabledThemes( const ThemeInfoList& themes, bool showAll );
|
||||||
|
/** @brief enable all installed plasma themes. */
|
||||||
|
void setEnabledThemesAll();
|
||||||
|
/** @brief set which theme is to be preselected. */
|
||||||
|
void setPreselect( const QString& id );
|
||||||
|
|
||||||
|
signals:
|
||||||
|
void plasmaThemeSelected( const QString& id );
|
||||||
|
|
||||||
|
private:
|
||||||
|
/** @brief Intersect the list of enabled themes with the installed ones. */
|
||||||
|
void winnowThemes();
|
||||||
|
/** @brief Get the translated names for all enabled themes. */
|
||||||
|
void updateThemeNames();
|
||||||
|
/** @brief show enabled themes in the UI. */
|
||||||
|
void fillUi();
|
||||||
|
|
||||||
|
Ui::PlasmaLnfPage* ui;
|
||||||
|
QString m_lnfPath;
|
||||||
|
QString m_preselect;
|
||||||
|
bool m_showAll; // If true, don't winnow according to enabledThemes
|
||||||
|
ThemeInfoList m_enabledThemes;
|
||||||
|
|
||||||
|
QButtonGroup *m_buttonGroup;
|
||||||
|
QList< ThemeWidget* > m_widgets;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif //PLASMALNFPAGE_H
|
225
src/modules/plasmalnf/PlasmaLnfViewStep.cpp
Normal file
225
src/modules/plasmalnf/PlasmaLnfViewStep.cpp
Normal file
@@ -0,0 +1,225 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2017, Adriaan de Groot <groot@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 "PlasmaLnfViewStep.h"
|
||||||
|
|
||||||
|
#include "PlasmaLnfJob.h"
|
||||||
|
#include "PlasmaLnfPage.h"
|
||||||
|
#include "ThemeInfo.h"
|
||||||
|
|
||||||
|
#include "utils/Logger.h"
|
||||||
|
|
||||||
|
#include <QProcess>
|
||||||
|
#include <QVariantMap>
|
||||||
|
|
||||||
|
#ifdef WITH_KCONFIG
|
||||||
|
#include <KConfigGroup>
|
||||||
|
#include <KSharedConfig>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
CALAMARES_PLUGIN_FACTORY_DEFINITION( PlasmaLnfViewStepFactory, registerPlugin<PlasmaLnfViewStep>(); )
|
||||||
|
|
||||||
|
static QString
|
||||||
|
currentPlasmaTheme()
|
||||||
|
{
|
||||||
|
#ifdef WITH_KCONFIG
|
||||||
|
KConfigGroup cg( KSharedConfig::openConfig( QStringLiteral( "kdeglobals" ) ), "KDE" );
|
||||||
|
return cg.readEntry( "LookAndFeelPackage", QString() );
|
||||||
|
#else
|
||||||
|
cWarning() << "No KConfig support, cannot determine Plasma theme.";
|
||||||
|
return QString();
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
PlasmaLnfViewStep::PlasmaLnfViewStep( QObject* parent )
|
||||||
|
: Calamares::ViewStep( parent )
|
||||||
|
, m_widget( new PlasmaLnfPage )
|
||||||
|
{
|
||||||
|
connect( m_widget, &PlasmaLnfPage::plasmaThemeSelected, this, &PlasmaLnfViewStep::themeSelected );
|
||||||
|
emit nextStatusChanged( false );
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
PlasmaLnfViewStep::~PlasmaLnfViewStep()
|
||||||
|
{
|
||||||
|
if ( m_widget && m_widget->parent() == nullptr )
|
||||||
|
m_widget->deleteLater();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
QString
|
||||||
|
PlasmaLnfViewStep::prettyName() const
|
||||||
|
{
|
||||||
|
return tr( "Look-and-Feel" );
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
QWidget*
|
||||||
|
PlasmaLnfViewStep::widget()
|
||||||
|
{
|
||||||
|
return m_widget;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void
|
||||||
|
PlasmaLnfViewStep::next()
|
||||||
|
{
|
||||||
|
emit done();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void
|
||||||
|
PlasmaLnfViewStep::back()
|
||||||
|
{}
|
||||||
|
|
||||||
|
|
||||||
|
bool
|
||||||
|
PlasmaLnfViewStep::isNextEnabled() const
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool
|
||||||
|
PlasmaLnfViewStep::isBackEnabled() const
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool
|
||||||
|
PlasmaLnfViewStep::isAtBeginning() const
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool
|
||||||
|
PlasmaLnfViewStep::isAtEnd() const
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void PlasmaLnfViewStep::onLeave()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
QList<Calamares::job_ptr>
|
||||||
|
PlasmaLnfViewStep::jobs() const
|
||||||
|
{
|
||||||
|
QList<Calamares::job_ptr> l;
|
||||||
|
|
||||||
|
cDebug() << "Creating Plasma LNF jobs ..";
|
||||||
|
if ( !m_themeId.isEmpty() )
|
||||||
|
{
|
||||||
|
if ( !m_lnfPath.isEmpty() )
|
||||||
|
l.append( Calamares::job_ptr( new PlasmaLnfJob( m_lnfPath, m_themeId ) ) );
|
||||||
|
else
|
||||||
|
cDebug() << "WARNING: no lnftool given for plasmalnf module.";
|
||||||
|
}
|
||||||
|
return l;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void
|
||||||
|
PlasmaLnfViewStep::setConfigurationMap( const QVariantMap& configurationMap )
|
||||||
|
{
|
||||||
|
QString lnfPath;
|
||||||
|
if ( configurationMap.contains( "lnftool" ) && configurationMap.value( "lnftool" ).type() == QVariant::String )
|
||||||
|
lnfPath = configurationMap.value( "lnftool" ).toString();
|
||||||
|
m_lnfPath = lnfPath;
|
||||||
|
m_widget->setLnfPath( m_lnfPath );
|
||||||
|
|
||||||
|
if ( m_lnfPath.isEmpty() )
|
||||||
|
cDebug() << "WARNING: no lnftool given for plasmalnf module.";
|
||||||
|
|
||||||
|
QString liveUser;
|
||||||
|
if ( configurationMap.contains( "liveuser" ) && configurationMap.value( "liveuser" ).type() == QVariant::String )
|
||||||
|
liveUser = configurationMap.value( "liveuser" ).toString();
|
||||||
|
m_liveUser = liveUser;
|
||||||
|
|
||||||
|
QString preselect;
|
||||||
|
if ( configurationMap.contains( "preselect" ) && configurationMap.value( "preselect" ).type() == QVariant::String )
|
||||||
|
preselect = configurationMap.value( "preselect" ).toString();
|
||||||
|
if ( preselect == QStringLiteral( "*" ) )
|
||||||
|
preselect = currentPlasmaTheme();
|
||||||
|
if ( !preselect.isEmpty() )
|
||||||
|
m_widget->setPreselect( preselect );
|
||||||
|
|
||||||
|
bool showAll( false );
|
||||||
|
if ( configurationMap.contains( "showAll" ) && configurationMap.value( "showAll" ).type() == QVariant::Bool )
|
||||||
|
showAll = configurationMap.value( "showAll" ).toBool();
|
||||||
|
|
||||||
|
if ( configurationMap.contains( "themes" ) &&
|
||||||
|
configurationMap.value( "themes" ).type() == QVariant::List )
|
||||||
|
{
|
||||||
|
ThemeInfoList listedThemes;
|
||||||
|
auto themeList = configurationMap.value( "themes" ).toList();
|
||||||
|
// Create the ThemInfo objects for the listed themes; information
|
||||||
|
// about the themes from Plasma (e.g. human-readable name and description)
|
||||||
|
// are filled in by update_names() in PlasmaLnfPage.
|
||||||
|
for ( const auto& i : themeList )
|
||||||
|
if ( i.type() == QVariant::Map )
|
||||||
|
{
|
||||||
|
auto iv = i.toMap();
|
||||||
|
listedThemes.append( ThemeInfo( iv.value( "theme" ).toString(), iv.value( "image" ).toString() ) );
|
||||||
|
}
|
||||||
|
else if ( i.type() == QVariant::String )
|
||||||
|
listedThemes.append( ThemeInfo( i.toString() ) );
|
||||||
|
|
||||||
|
if ( listedThemes.length() == 1 )
|
||||||
|
cDebug() << "WARNING: only one theme enabled in plasmalnf";
|
||||||
|
m_widget->setEnabledThemes( listedThemes, showAll );
|
||||||
|
}
|
||||||
|
else
|
||||||
|
m_widget->setEnabledThemesAll(); // All of them
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
PlasmaLnfViewStep::themeSelected( const QString& id )
|
||||||
|
{
|
||||||
|
m_themeId = id;
|
||||||
|
if ( m_lnfPath.isEmpty() )
|
||||||
|
{
|
||||||
|
cDebug() << "WARNING: no lnftool given for plasmalnf module.";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
QProcess lnftool;
|
||||||
|
if ( !m_liveUser.isEmpty() )
|
||||||
|
lnftool.start( "sudo", {"-E", "-H", "-u", m_liveUser, m_lnfPath, "--resetLayout", "--apply", id} );
|
||||||
|
else
|
||||||
|
lnftool.start( m_lnfPath, {"--resetLayout", "--apply", id} );
|
||||||
|
|
||||||
|
if ( !lnftool.waitForStarted( 1000 ) )
|
||||||
|
{
|
||||||
|
cDebug() << "WARNING: could not start look-and-feel" << m_lnfPath;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if ( !lnftool.waitForFinished() )
|
||||||
|
{
|
||||||
|
cDebug() << "WARNING:" << m_lnfPath << "timed out.";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( ( lnftool.exitCode() == 0 ) && ( lnftool.exitStatus() == QProcess::NormalExit ) )
|
||||||
|
cDebug() << "Plasma look-and-feel applied" << id;
|
||||||
|
else
|
||||||
|
cDebug() << "WARNING: could not apply look-and-feel" << id;
|
||||||
|
}
|
71
src/modules/plasmalnf/PlasmaLnfViewStep.h
Normal file
71
src/modules/plasmalnf/PlasmaLnfViewStep.h
Normal file
@@ -0,0 +1,71 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2017, Adriaan de Groot <groot@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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef PLASMALNFVIEWSTEP_H
|
||||||
|
#define PLASMALNFVIEWSTEP_H
|
||||||
|
|
||||||
|
#include <utils/PluginFactory.h>
|
||||||
|
#include <viewpages/ViewStep.h>
|
||||||
|
#include <PluginDllMacro.h>
|
||||||
|
|
||||||
|
#include <QObject>
|
||||||
|
#include <QUrl>
|
||||||
|
#include <QVariantMap>
|
||||||
|
|
||||||
|
class PlasmaLnfPage;
|
||||||
|
|
||||||
|
class PLUGINDLLEXPORT PlasmaLnfViewStep : public Calamares::ViewStep
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit PlasmaLnfViewStep( QObject* parent = nullptr );
|
||||||
|
virtual ~PlasmaLnfViewStep() override;
|
||||||
|
|
||||||
|
QString prettyName() const override;
|
||||||
|
|
||||||
|
QWidget* widget() override;
|
||||||
|
|
||||||
|
void next() override;
|
||||||
|
void back() override;
|
||||||
|
|
||||||
|
bool isNextEnabled() const override;
|
||||||
|
bool isBackEnabled() const override;
|
||||||
|
|
||||||
|
bool isAtBeginning() const override;
|
||||||
|
bool isAtEnd() const override;
|
||||||
|
|
||||||
|
void onLeave() override;
|
||||||
|
|
||||||
|
QList<Calamares::job_ptr> jobs() const override;
|
||||||
|
|
||||||
|
void setConfigurationMap( const QVariantMap& configurationMap ) override;
|
||||||
|
|
||||||
|
public slots:
|
||||||
|
void themeSelected( const QString& id );
|
||||||
|
|
||||||
|
private:
|
||||||
|
PlasmaLnfPage* m_widget;
|
||||||
|
QString m_lnfPath; // Path to the lnf tool
|
||||||
|
QString m_themeId; // Id of selected theme
|
||||||
|
QString m_liveUser; // Name of the live user (for OEM mode)
|
||||||
|
};
|
||||||
|
|
||||||
|
CALAMARES_PLUGIN_FACTORY_DECLARATION( PlasmaLnfViewStepFactory )
|
||||||
|
|
||||||
|
#endif // PLASMALNFVIEWSTEP_H
|
97
src/modules/plasmalnf/ThemeInfo.h
Normal file
97
src/modules/plasmalnf/ThemeInfo.h
Normal file
@@ -0,0 +1,97 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2017, Adriaan de Groot <groot@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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef PLASMALNF_THEMEINFO_H
|
||||||
|
#define PLASMALNF_THEMEINFO_H
|
||||||
|
|
||||||
|
#include <QList>
|
||||||
|
#include <QString>
|
||||||
|
|
||||||
|
class KPluginMetaData;
|
||||||
|
class ThemeWidget;
|
||||||
|
|
||||||
|
/** @brief describes a single plasma LnF theme.
|
||||||
|
*
|
||||||
|
* A theme description has an id, which is really the name of the desktop
|
||||||
|
* file (e.g. org.kde.breeze.desktop), a name which is human-readable and
|
||||||
|
* translated, and an optional image Page, which points to a local screenshot
|
||||||
|
* of that theme.
|
||||||
|
*/
|
||||||
|
struct ThemeInfo
|
||||||
|
{
|
||||||
|
QString id;
|
||||||
|
QString name;
|
||||||
|
QString description;
|
||||||
|
QString imagePath;
|
||||||
|
ThemeWidget* widget;
|
||||||
|
|
||||||
|
ThemeInfo()
|
||||||
|
: widget( nullptr )
|
||||||
|
{}
|
||||||
|
|
||||||
|
explicit ThemeInfo( const QString& _id )
|
||||||
|
: id( _id )
|
||||||
|
, widget( nullptr )
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
explicit ThemeInfo( const QString& _id, const QString& image )
|
||||||
|
: id( _id )
|
||||||
|
, imagePath( image )
|
||||||
|
, widget( nullptr )
|
||||||
|
{}
|
||||||
|
|
||||||
|
// Defined in PlasmaLnfPage.cpp
|
||||||
|
explicit ThemeInfo( const KPluginMetaData& );
|
||||||
|
|
||||||
|
bool isValid() const { return !id.isEmpty(); }
|
||||||
|
} ;
|
||||||
|
|
||||||
|
class ThemeInfoList : public QList< ThemeInfo >
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
/** @brief Looks for a given @p id in the list of themes, returns nullptr if not found. */
|
||||||
|
ThemeInfo* findById( const QString& id )
|
||||||
|
{
|
||||||
|
for ( ThemeInfo& i : *this )
|
||||||
|
{
|
||||||
|
if ( i.id == id )
|
||||||
|
return &i;
|
||||||
|
}
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @brief Looks for a given @p id in the list of themes, returns nullptr if not found. */
|
||||||
|
const ThemeInfo* findById( const QString& id ) const
|
||||||
|
{
|
||||||
|
for ( const ThemeInfo& i : *this )
|
||||||
|
{
|
||||||
|
if ( i.id == id )
|
||||||
|
return &i;
|
||||||
|
}
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @brief Checks if a given @p id is in the list of themes. */
|
||||||
|
bool contains( const QString& id ) const
|
||||||
|
{
|
||||||
|
return findById( id ) != nullptr;
|
||||||
|
}
|
||||||
|
} ;
|
||||||
|
|
||||||
|
#endif
|
85
src/modules/plasmalnf/ThemeWidget.cpp
Normal file
85
src/modules/plasmalnf/ThemeWidget.cpp
Normal file
@@ -0,0 +1,85 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2017, Adriaan de Groot <groot@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 "ThemeWidget.h"
|
||||||
|
|
||||||
|
#include "ThemeInfo.h"
|
||||||
|
|
||||||
|
#include "utils/Logger.h"
|
||||||
|
|
||||||
|
#include <QHBoxLayout>
|
||||||
|
#include <QLabel>
|
||||||
|
#include <QRadioButton>
|
||||||
|
|
||||||
|
ThemeWidget::ThemeWidget(const ThemeInfo& info, QWidget* parent)
|
||||||
|
: QWidget( parent )
|
||||||
|
, m_check( new QRadioButton( info.name.isEmpty() ? info.id : info.name, parent ) )
|
||||||
|
, m_description( new QLabel( info.description, parent ) )
|
||||||
|
, m_id( info.id )
|
||||||
|
{
|
||||||
|
QHBoxLayout* layout = new QHBoxLayout( this );
|
||||||
|
this->setLayout( layout );
|
||||||
|
|
||||||
|
layout->addWidget( m_check, 1 );
|
||||||
|
|
||||||
|
constexpr QSize image_size{120, 80};
|
||||||
|
|
||||||
|
QPixmap image( info.imagePath );
|
||||||
|
if ( info.imagePath.isEmpty() )
|
||||||
|
{
|
||||||
|
// Image can't possibly be valid
|
||||||
|
image = QPixmap( ":/view-preview.png" );
|
||||||
|
}
|
||||||
|
else if ( image.isNull() )
|
||||||
|
{
|
||||||
|
// Not found or not specified, so convert the name into some (horrible, likely)
|
||||||
|
// color instead.
|
||||||
|
image = QPixmap( image_size );
|
||||||
|
uint hash_color = qHash( info.imagePath.isEmpty() ? info.id : info.imagePath );
|
||||||
|
cDebug() << "Theme image" << info.imagePath << "not found, hash" << hash_color;
|
||||||
|
image.fill( QColor( QRgb( hash_color ) ) );
|
||||||
|
}
|
||||||
|
else
|
||||||
|
image.scaled( image_size );
|
||||||
|
|
||||||
|
QLabel* image_label = new QLabel( this );
|
||||||
|
image_label->setPixmap( image );
|
||||||
|
layout->addWidget( image_label, 1 );
|
||||||
|
layout->addWidget( m_description, 3 );
|
||||||
|
|
||||||
|
connect( m_check, &QRadioButton::toggled, this, &ThemeWidget::clicked );
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
ThemeWidget::clicked( bool checked )
|
||||||
|
{
|
||||||
|
if ( checked )
|
||||||
|
emit themeSelected( m_id );
|
||||||
|
}
|
||||||
|
|
||||||
|
QAbstractButton*
|
||||||
|
ThemeWidget::button() const
|
||||||
|
{
|
||||||
|
return m_check;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ThemeWidget::updateThemeName(const ThemeInfo& info)
|
||||||
|
{
|
||||||
|
m_check->setText( info.name );
|
||||||
|
m_description->setText( info.description );
|
||||||
|
}
|
53
src/modules/plasmalnf/ThemeWidget.h
Normal file
53
src/modules/plasmalnf/ThemeWidget.h
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
||||||
|
*
|
||||||
|
* Copyright 2017, Adriaan de Groot <groot@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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef PLASMALNF_THEMEWIDGET_H
|
||||||
|
#define PLASMALNF_THEMEWIDGET_H
|
||||||
|
|
||||||
|
#include <QWidget>
|
||||||
|
|
||||||
|
class QAbstractButton;
|
||||||
|
class QLabel;
|
||||||
|
class QRadioButton;
|
||||||
|
|
||||||
|
struct ThemeInfo;
|
||||||
|
|
||||||
|
class ThemeWidget : public QWidget
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
public:
|
||||||
|
explicit ThemeWidget( const ThemeInfo& info, QWidget* parent = nullptr );
|
||||||
|
|
||||||
|
QAbstractButton* button() const;
|
||||||
|
|
||||||
|
void updateThemeName( const ThemeInfo& info );
|
||||||
|
|
||||||
|
signals:
|
||||||
|
void themeSelected( const QString& id );
|
||||||
|
|
||||||
|
public slots:
|
||||||
|
void clicked( bool );
|
||||||
|
|
||||||
|
private:
|
||||||
|
QString m_id;
|
||||||
|
QRadioButton* m_check;
|
||||||
|
QLabel* m_description;
|
||||||
|
} ;
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
5
src/modules/plasmalnf/page_plasmalnf.qrc
Normal file
5
src/modules/plasmalnf/page_plasmalnf.qrc
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
<RCC>
|
||||||
|
<qresource>
|
||||||
|
<file>view-preview.png</file>
|
||||||
|
</qresource>
|
||||||
|
</RCC>
|
46
src/modules/plasmalnf/page_plasmalnf.ui
Normal file
46
src/modules/plasmalnf/page_plasmalnf.ui
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<ui version="4.0">
|
||||||
|
<class>PlasmaLnfPage</class>
|
||||||
|
<widget class="QWidget" name="PlasmaLnfPage">
|
||||||
|
<property name="geometry">
|
||||||
|
<rect>
|
||||||
|
<x>0</x>
|
||||||
|
<y>0</y>
|
||||||
|
<width>799</width>
|
||||||
|
<height>400</height>
|
||||||
|
</rect>
|
||||||
|
</property>
|
||||||
|
<property name="windowTitle">
|
||||||
|
<string>Form</string>
|
||||||
|
</property>
|
||||||
|
<layout class="QVBoxLayout" name="verticalLayout" stretch="0,0">
|
||||||
|
<item>
|
||||||
|
<widget class="QLabel" name="generalExplanation">
|
||||||
|
<property name="text">
|
||||||
|
<string>Placeholder</string>
|
||||||
|
</property>
|
||||||
|
<property name="wordWrap">
|
||||||
|
<bool>true</bool>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<spacer name="verticalSpacer">
|
||||||
|
<property name="orientation">
|
||||||
|
<enum>Qt::Vertical</enum>
|
||||||
|
</property>
|
||||||
|
<property name="sizeHint" stdset="0">
|
||||||
|
<size>
|
||||||
|
<width>20</width>
|
||||||
|
<height>40</height>
|
||||||
|
</size>
|
||||||
|
</property>
|
||||||
|
</spacer>
|
||||||
|
</item>
|
||||||
|
</layout>
|
||||||
|
</widget>
|
||||||
|
<resources>
|
||||||
|
<include location="page_plasmalnf.qrc"/>
|
||||||
|
</resources>
|
||||||
|
<connections/>
|
||||||
|
</ui>
|
53
src/modules/plasmalnf/plasmalnf.conf
Normal file
53
src/modules/plasmalnf/plasmalnf.conf
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
---
|
||||||
|
# Full path to the Plasma look-and-feel tool (CLI program
|
||||||
|
# for querying and applying Plasma themes). If this is not
|
||||||
|
# set, no LNF setting will happen.
|
||||||
|
lnftool: "/usr/bin/lookandfeeltool"
|
||||||
|
|
||||||
|
# For systems where the user Calamares runs as (usually root,
|
||||||
|
# via either sudo or pkexec) has a clean environment, set this
|
||||||
|
# to the originating username; the lnftool will be run through
|
||||||
|
# "sudo -H -u <liveuser>" instead of directly.
|
||||||
|
#
|
||||||
|
# liveuser: "live"
|
||||||
|
|
||||||
|
# You can limit the list of Plasma look-and-feel themes by listing ids
|
||||||
|
# here. If this key is not present, all of the installed themes are listed.
|
||||||
|
# If the key is present, only installed themes that are **also** included
|
||||||
|
# in the list are shown (could be none!). See the *showAll* key, below,
|
||||||
|
# to change that.
|
||||||
|
#
|
||||||
|
# Themes may be listed by id, (e.g. fluffy-bunny, below) or as a theme
|
||||||
|
# and an image (e.g. breeze) which will be used to show a screenshot.
|
||||||
|
# Themes with no image get a "missing screenshot" image; if the
|
||||||
|
# image file is not found, they get a color swatch based on the image name.
|
||||||
|
themes:
|
||||||
|
- org.kde.fuzzy-pig.desktop
|
||||||
|
- theme: org.kde.breeze.desktop
|
||||||
|
image: "breeze.png"
|
||||||
|
- theme: org.kde.breezedark.desktop
|
||||||
|
image: "breeze-dark.png"
|
||||||
|
- org.kde.fluffy-bunny.desktop
|
||||||
|
|
||||||
|
# If *showAll* is true, then all installed themes are shown in the
|
||||||
|
# UI for selection, even if they are not listed in *themes*. This
|
||||||
|
# allows selection of all themes even while not all of them are
|
||||||
|
# listed in *themes* -- which is useful to show screenshots for those
|
||||||
|
# you do have a screenshot for.
|
||||||
|
showAll: false
|
||||||
|
|
||||||
|
# You can pre-select one of the themes; it is not applied
|
||||||
|
# immediately, but its radio-button is switched on to indicate
|
||||||
|
# that that is the theme (that is most likely) currently in use.
|
||||||
|
# Do this only on Live images where you are reasonably sure
|
||||||
|
# that the user is not going to change the theme out from under
|
||||||
|
# themselves before running the installer.
|
||||||
|
#
|
||||||
|
# If this key is present, its value should be the id of the theme
|
||||||
|
# which should be pre-selected. If absent, empty, or the pre-selected
|
||||||
|
# theme is not found on the live system, no theme will be pre-selected.
|
||||||
|
#
|
||||||
|
# As a special setting, use "*", to try to find the currently-
|
||||||
|
# selected theme by reading the Plasma configuration. This requires
|
||||||
|
# KF5::Config at build- and run-time.
|
||||||
|
preselect: "*"
|
BIN
src/modules/plasmalnf/view-preview.png
Normal file
BIN
src/modules/plasmalnf/view-preview.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 560 B |
13
src/modules/plasmalnf/view-preview.svg
Normal file
13
src/modules/plasmalnf/view-preview.svg
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 32 32">
|
||||||
|
<defs id="defs3051">
|
||||||
|
<style type="text/css" id="current-color-scheme">
|
||||||
|
.ColorScheme-Text {
|
||||||
|
color:#4d4d4d;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</defs>
|
||||||
|
<path style="fill:currentColor;fill-opacity:1;stroke:none"
|
||||||
|
d="m4 4v24h24v-24zm1 1h22v22h-22zm6 2a4 4 0 0 0 -4 4 4 4 0 0 0 4 4 4 4 0 0 0 4 -4 4 4 0 0 0 -4 -4m0 1a3 3 0 0 1 3 3 3 3 0 0 1 -3 3 3 3 0 0 1 -3 -3 3 3 0 0 1 3 -3m9.5 6.793l-5 5-2-2-6.5 6.5.707.707 5.793-5.793 2 2 5-5 3.793 3.793.707-.707z"
|
||||||
|
class="ColorScheme-Text"
|
||||||
|
/>
|
||||||
|
</svg>
|
After Width: | Height: | Size: 570 B |
@@ -1,6 +1,5 @@
|
|||||||
find_package(ECM 5.10.0 NO_MODULE)
|
find_package(ECM ${ECM_VERSION} NO_MODULE)
|
||||||
if( ECM_FOUND )
|
if( ECM_FOUND )
|
||||||
set(CMAKE_MODULE_PATH ${ECM_MODULE_PATH} ${CMAKE_MODULE_PATH})
|
|
||||||
include( ECMAddTests )
|
include( ECMAddTests )
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
|
@@ -213,12 +213,16 @@ RequirementsChecker::setConfigurationMap( const QVariantMap& configurationMap )
|
|||||||
bool ok = false;
|
bool ok = false;
|
||||||
m_requiredStorageGB = configurationMap.value( "requiredStorage" ).toDouble( &ok );
|
m_requiredStorageGB = configurationMap.value( "requiredStorage" ).toDouble( &ok );
|
||||||
if ( !ok )
|
if ( !ok )
|
||||||
|
{
|
||||||
|
cDebug() << "WARNING: RequirementsChecker entry 'requiredStorage' is invalid.";
|
||||||
m_requiredStorageGB = 3.;
|
m_requiredStorageGB = 3.;
|
||||||
|
}
|
||||||
|
|
||||||
Calamares::JobQueue::instance()->globalStorage()->insert( "requiredStorageGB", m_requiredStorageGB );
|
Calamares::JobQueue::instance()->globalStorage()->insert( "requiredStorageGB", m_requiredStorageGB );
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
cDebug() << "WARNING: RequirementsChecker entry 'requiredStorage' is missing.";
|
||||||
m_requiredStorageGB = 3.;
|
m_requiredStorageGB = 3.;
|
||||||
incompleteConfiguration = true;
|
incompleteConfiguration = true;
|
||||||
}
|
}
|
||||||
@@ -231,12 +235,14 @@ RequirementsChecker::setConfigurationMap( const QVariantMap& configurationMap )
|
|||||||
m_requiredRamGB = configurationMap.value( "requiredRam" ).toDouble( &ok );
|
m_requiredRamGB = configurationMap.value( "requiredRam" ).toDouble( &ok );
|
||||||
if ( !ok )
|
if ( !ok )
|
||||||
{
|
{
|
||||||
|
cDebug() << "WARNING: RequirementsChecker entry 'requiredRam' is invalid.";
|
||||||
m_requiredRamGB = 1.;
|
m_requiredRamGB = 1.;
|
||||||
incompleteConfiguration = true;
|
incompleteConfiguration = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
cDebug() << "WARNING: RequirementsChecker entry 'requiredRam' is missing.";
|
||||||
m_requiredRamGB = 1.;
|
m_requiredRamGB = 1.;
|
||||||
incompleteConfiguration = true;
|
incompleteConfiguration = true;
|
||||||
}
|
}
|
||||||
@@ -248,7 +254,7 @@ RequirementsChecker::setConfigurationMap( const QVariantMap& configurationMap )
|
|||||||
if ( m_checkHasInternetUrl.isEmpty() ||
|
if ( m_checkHasInternetUrl.isEmpty() ||
|
||||||
!QUrl( m_checkHasInternetUrl ).isValid() )
|
!QUrl( m_checkHasInternetUrl ).isValid() )
|
||||||
{
|
{
|
||||||
cDebug() << "Invalid internetCheckUrl in welcome.conf" << m_checkHasInternetUrl
|
cDebug() << "WARNING: RequirementsChecker entry 'internetCheckUrl' is invalid in welcome.conf" << m_checkHasInternetUrl
|
||||||
<< "reverting to default (http://example.com).";
|
<< "reverting to default (http://example.com).";
|
||||||
m_checkHasInternetUrl = "http://example.com";
|
m_checkHasInternetUrl = "http://example.com";
|
||||||
incompleteConfiguration = true;
|
incompleteConfiguration = true;
|
||||||
@@ -256,8 +262,9 @@ RequirementsChecker::setConfigurationMap( const QVariantMap& configurationMap )
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
cDebug() << "internetCheckUrl is undefined in welcome.conf, "
|
cDebug() << "WARNING: RequirementsChecker entry 'internetCheckUrl' is undefined in welcome.conf,"
|
||||||
"reverting to default (http://example.com).";
|
"reverting to default (http://example.com).";
|
||||||
|
|
||||||
m_checkHasInternetUrl = "http://example.com";
|
m_checkHasInternetUrl = "http://example.com";
|
||||||
incompleteConfiguration = true;
|
incompleteConfiguration = true;
|
||||||
}
|
}
|
||||||
@@ -269,7 +276,10 @@ RequirementsChecker::setConfigurationMap( const QVariantMap& configurationMap )
|
|||||||
m_entriesToCheck.append( configurationMap.value( "check" ).toStringList() );
|
m_entriesToCheck.append( configurationMap.value( "check" ).toStringList() );
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
{
|
||||||
|
cDebug() << "WARNING: RequirementsChecker entry 'check' is incomplete.";
|
||||||
incompleteConfiguration = true;
|
incompleteConfiguration = true;
|
||||||
|
}
|
||||||
|
|
||||||
if ( configurationMap.contains( "required" ) &&
|
if ( configurationMap.contains( "required" ) &&
|
||||||
configurationMap.value( "required" ).type() == QVariant::List )
|
configurationMap.value( "required" ).type() == QVariant::List )
|
||||||
@@ -278,18 +288,13 @@ RequirementsChecker::setConfigurationMap( const QVariantMap& configurationMap )
|
|||||||
m_entriesToRequire.append( configurationMap.value( "required" ).toStringList() );
|
m_entriesToRequire.append( configurationMap.value( "required" ).toStringList() );
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
{
|
||||||
|
cDebug() << "WARNING: RequirementsChecker entry 'required' is incomplete.";
|
||||||
incompleteConfiguration = true;
|
incompleteConfiguration = true;
|
||||||
|
}
|
||||||
|
|
||||||
if ( incompleteConfiguration )
|
if ( incompleteConfiguration )
|
||||||
{
|
cDebug() << "WARNING: RequirementsChecker configuration map:\n" << configurationMap;
|
||||||
cDebug() << "WARNING: The RequirementsChecker configuration map provided by "
|
|
||||||
"the welcome module configuration file is incomplete or "
|
|
||||||
"incorrect.\n"
|
|
||||||
"Startup will continue for debugging purposes, but one or "
|
|
||||||
"more checks might not function correctly.\n"
|
|
||||||
"RequirementsChecker configuration map:\n"
|
|
||||||
<< configurationMap;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@@ -25,6 +25,6 @@ requirements:
|
|||||||
# If any of these conditions are not met, the user cannot
|
# If any of these conditions are not met, the user cannot
|
||||||
# continue past the welcome page.
|
# continue past the welcome page.
|
||||||
required:
|
required:
|
||||||
- storage
|
# - storage
|
||||||
- ram
|
- ram
|
||||||
- root
|
# - root
|
||||||
|
Reference in New Issue
Block a user