/*************************************************************************** * Copyright (c) 2004 Werner Mayer * * * * This file is part of the FreeCAD CAx development system. * * * * This library is free software; you can redistribute it and/or * * modify it under the terms of the GNU Library General Public * * License as published by the Free Software Foundation; either * * version 2 of the License, or (at your option) any later version. * * * * This library 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 Library General Public License for more details. * * * * You should have received a copy of the GNU Library General Public * * License along with this library; see the file COPYING.LIB. If not, * * write to the Free Software Foundation, Inc., 59 Temple Place, * * Suite 330, Boston, MA 02111-1307, USA * * * ***************************************************************************/ #include "PreCompiled.h" #ifndef _PreComp_ # include # include # include # include # include # include # include # include # include # include # include # include # include # include # include # include # include #endif #include #include #include #include #include #include #include #include "Splashscreen.h" #include "ui_AboutApplication.h" #include "MainWindow.h" using namespace Gui; using namespace Gui::Dialog; namespace fs = boost::filesystem; namespace Gui { QString prettyProductInfoWrapper() { auto productName = QSysInfo::prettyProductName(); #if QT_VERSION < QT_VERSION_CHECK(6, 5, 0) #ifdef FC_OS_MACOSX auto macosVersionFile = QString::fromUtf8("/System/Library/CoreServices/.SystemVersionPlatform.plist"); auto fi = QFileInfo (macosVersionFile); if (fi.exists() && fi.isReadable()) { auto plistFile = QFile(macosVersionFile); plistFile.open(QIODevice::ReadOnly); while (!plistFile.atEnd()) { auto line = plistFile.readLine(); if (line.contains("ProductUserVisibleVersion")) { auto nextLine = plistFile.readLine(); if (nextLine.contains("")) { QRegularExpression re(QString::fromUtf8("\\s*(.*)")); auto matches = re.match(QString::fromUtf8(nextLine)); if (matches.hasMatch()) { productName = QString::fromUtf8("macOS ") + matches.captured(1); break; } } } } } #endif #endif #ifdef FC_OS_WIN64 QSettings regKey {QString::fromUtf8("HKEY_LOCAL_MACHINE\\SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion"), QSettings::NativeFormat}; if (regKey.contains(QString::fromUtf8("CurrentBuildNumber"))) { auto buildNumber = regKey.value(QString::fromUtf8("CurrentBuildNumber")).toInt(); if (buildNumber > 0) { if (buildNumber < 9200) { productName = QString::fromUtf8("Windows 7 build %1").arg(buildNumber); } else if (buildNumber < 10240) { productName = QString::fromUtf8("Windows 8 build %1").arg(buildNumber); } else if (buildNumber < 22000) { productName = QString::fromUtf8("Windows 10 build %1").arg(buildNumber); } else { productName = QString::fromUtf8("Windows 11 build %1").arg(buildNumber); } } } #endif return productName; } /** Displays all messages at startup inside the splash screen. * \author Werner Mayer */ class SplashObserver : public Base::ILogger { public: SplashObserver(const SplashObserver&) = delete; SplashObserver(SplashObserver&&) = delete; SplashObserver& operator= (const SplashObserver&) = delete; SplashObserver& operator= (SplashObserver&&) = delete; explicit SplashObserver(QSplashScreen* splasher=nullptr) : splash(splasher) , alignment(Qt::AlignBottom|Qt::AlignLeft) , textColor(Qt::black) { Base::Console().AttachObserver(this); // allow to customize text position and color const std::map& cfg = App::Application::Config(); auto al = cfg.find("SplashAlignment"); if (al != cfg.end()) { QString alt = QString::fromLatin1(al->second.c_str()); int align=0; if (alt.startsWith(QLatin1String("VCenter"))) { align = Qt::AlignVCenter; } else if (alt.startsWith(QLatin1String("Top"))) { align = Qt::AlignTop; } else { align = Qt::AlignBottom; } if (alt.endsWith(QLatin1String("HCenter"))) { align += Qt::AlignHCenter; } else if (alt.endsWith(QLatin1String("Right"))) { align += Qt::AlignRight; } else { align += Qt::AlignLeft; } alignment = align; } // choose text color auto tc = cfg.find("SplashTextColor"); if (tc != cfg.end()) { QColor col(QString::fromStdString(tc->second)); if (col.isValid()) { textColor = col; } } } ~SplashObserver() override { Base::Console().DetachObserver(this); } const char* Name() override { return "SplashObserver"; } void SendLog(const std::string& notifiername, const std::string& msg, Base::LogStyle level, Base::IntendedRecipient recipient, Base::ContentType content) override { Q_UNUSED(notifiername) Q_UNUSED(recipient) Q_UNUSED(content) #ifdef FC_DEBUG Q_UNUSED(level) Log(msg); #else if (level == Base::LogStyle::Log) { Log(msg); } #endif } void Log(const std::string& text) { QString msg(QString::fromStdString(text)); QRegularExpression rx; // ignore 'Init:' and 'Mod:' prefixes rx.setPattern(QLatin1String("^\\s*(Init:|Mod:)\\s*")); auto match = rx.match(msg); if (match.hasMatch()) { msg = msg.mid(match.capturedLength()); } else { // ignore activation of commands rx.setPattern(QLatin1String(R"(^\s*(\+App::|Create|CmdC:|CmdG:|Act:)\s*)")); match = rx.match(msg); if (match.hasMatch() && match.capturedStart() == 0) return; } splash->showMessage(msg.replace(QLatin1String("\n"), QString()), alignment, textColor); QMutex mutex; QMutexLocker ml(&mutex); QWaitCondition().wait(&mutex, 50); } private: QSplashScreen* splash; int alignment; QColor textColor; }; } // namespace Gui // ------------------------------------------------------------------------------ /** * Constructs a splash screen that will display the pixmap. */ SplashScreen::SplashScreen( const QPixmap & pixmap , Qt::WindowFlags f ) : QSplashScreen(pixmap, f) { // write the messages to splasher messages = new SplashObserver(this); } /** Destruction. */ SplashScreen::~SplashScreen() { delete messages; } /** * Draws the contents of the splash screen using painter \a painter. The default * implementation draws the message passed by message(). */ void SplashScreen::drawContents ( QPainter * painter ) { QSplashScreen::drawContents(painter); } void SplashScreen::setShowMessages(bool on) { messages->bErr = on; messages->bMsg = on; messages->bLog = on; messages->bWrn = on; } // ------------------------------------------------------------------------------ AboutDialogFactory* AboutDialogFactory::factory = nullptr; AboutDialogFactory::~AboutDialogFactory() = default; QDialog *AboutDialogFactory::create(QWidget *parent) const { return new AboutDialog(parent); } const AboutDialogFactory *AboutDialogFactory::defaultFactory() { static const AboutDialogFactory this_factory; if (factory) return factory; return &this_factory; } void AboutDialogFactory::setDefaultFactory(AboutDialogFactory *f) { if (factory != f) delete factory; factory = f; } // ------------------------------------------------------------------------------ /* TRANSLATOR Gui::Dialog::AboutDialog */ /** * Constructs an AboutDialog which is a child of 'parent', with the * name 'name' and widget flags set to 'WStyle_Customize|WStyle_NoBorder|WType_Modal' * * The dialog will be modal. */ AboutDialog::AboutDialog(QWidget* parent) : QDialog(parent), ui(new Ui_AboutApplication) { setModal(true); ui->setupUi(this); connect(ui->copyButton, &QPushButton::clicked, this, &AboutDialog::copyToClipboard); // remove the automatic help button in dialog title since we don't use it setWindowFlag(Qt::WindowContextHelpButtonHint, false); layout()->setSizeConstraint(QLayout::SetFixedSize); QRect rect = QApplication::primaryScreen()->availableGeometry(); // See if we have a custom About screen image set QPixmap image = getMainWindow()->aboutImage(); // Fallback to the splashscreen image if (image.isNull()) { image = getMainWindow()->splashImage(); } // Make sure the image is not too big int denom = 2; if (image.height() > rect.height()/denom || image.width() > rect.width()/denom) { float scale = static_cast(image.width()) / static_cast(image.height()); int width = std::min(image.width(), rect.width()/denom); int height = std::min(image.height(), rect.height()/denom); height = std::min(height, static_cast(width / scale)); width = static_cast(scale * height); image = image.scaled(width, height); } ui->labelSplashPicture->setPixmap(image); ui->tabWidget->setCurrentIndex(0); // always start on the About tab setupLabels(); showCredits(); showLicenseInformation(); showLibraryInformation(); showCollectionInformation(); showPrivacyPolicy(); showOrHideImage(rect); } /** * Destroys the object and frees any allocated resources */ AboutDialog::~AboutDialog() { // no need to delete child widgets, Qt does it all for us delete ui; } void AboutDialog::showOrHideImage(const QRect& rect) { adjustSize(); if (height() > rect.height()) { ui->labelSplashPicture->hide(); } } void AboutDialog::setupLabels() { //fonts are rendered smaller on Mac so point size can't be the same for all platforms int fontSize = 8; #ifdef Q_OS_MAC fontSize = 11; #endif //avoid overriding user set style sheet if (qApp->styleSheet().isEmpty()) { setStyleSheet(QString::fromLatin1("Gui--Dialog--AboutDialog QLabel {font-size: %1pt;}").arg(fontSize)); } QString exeName = qApp->applicationName(); std::map& config = App::Application::Config(); std::map::iterator it; QString banner = QString::fromUtf8(config["CopyrightInfo"].c_str()); banner = banner.left( banner.indexOf(QLatin1Char('\n')) ); QString major = QString::fromLatin1(config["BuildVersionMajor"].c_str()); QString minor = QString::fromLatin1(config["BuildVersionMinor"].c_str()); QString point = QString::fromLatin1(config["BuildVersionPoint"].c_str()); QString suffix = QString::fromLatin1(config["BuildVersionSuffix"].c_str()); QString build = QString::fromLatin1(config["BuildRevision"].c_str()); QString disda = QString::fromLatin1(config["BuildRevisionDate"].c_str()); QString mturl = QString::fromLatin1(config["MaintainerUrl"].c_str()); // we use replace() to keep label formatting, so a label with text "Unknown" // gets replaced to "FreeCAD", for example QString author = ui->labelAuthor->text(); author.replace(QString::fromLatin1("Unknown Application"), exeName); author.replace(QString::fromLatin1("(c) Unknown Author"), banner); ui->labelAuthor->setText(author); ui->labelAuthor->setUrl(mturl); if (qApp->styleSheet().isEmpty()) { ui->labelAuthor->setStyleSheet(QString::fromLatin1("Gui--UrlLabel {color: #0000FF;text-decoration: underline;font-weight: 600;}")); } QString version = ui->labelBuildVersion->text(); version.replace(QString::fromLatin1("Unknown"), QString::fromLatin1("%1.%2.%3%4").arg(major, minor, point, suffix)); ui->labelBuildVersion->setText(version); QString revision = ui->labelBuildRevision->text(); revision.replace(QString::fromLatin1("Unknown"), build); ui->labelBuildRevision->setText(revision); QString date = ui->labelBuildDate->text(); date.replace(QString::fromLatin1("Unknown"), disda); ui->labelBuildDate->setText(date); QString os = ui->labelBuildOS->text(); os.replace(QString::fromLatin1("Unknown"), prettyProductInfoWrapper()); ui->labelBuildOS->setText(os); QString architecture = ui->labelBuildRunArchitecture->text(); if (QSysInfo::buildCpuArchitecture() == QSysInfo::currentCpuArchitecture()) { architecture.replace(QString::fromLatin1("Unknown"), QSysInfo::buildCpuArchitecture()); } else { architecture.replace( QString::fromLatin1("Unknown"), QString::fromLatin1("%1 (running on: %2)") .arg(QSysInfo::buildCpuArchitecture(), QSysInfo::currentCpuArchitecture())); } ui->labelBuildRunArchitecture->setText(architecture); // branch name it = config.find("BuildRevisionBranch"); if (it != config.end()) { QString branch = ui->labelBuildBranch->text(); branch.replace(QString::fromLatin1("Unknown"), QString::fromUtf8(it->second.c_str())); ui->labelBuildBranch->setText(branch); } else { ui->labelBranch->hide(); ui->labelBuildBranch->hide(); } // hash id it = config.find("BuildRevisionHash"); if (it != config.end()) { QString hash = ui->labelBuildHash->text(); hash.replace(QString::fromLatin1("Unknown"), QString::fromLatin1(it->second.c_str()).left(7)); // Use the 7-char abbreviated hash ui->labelBuildHash->setText(hash); if (auto url_itr = config.find("BuildRepositoryURL"); url_itr != config.end()) { auto url = QString::fromStdString(url_itr->second); if (int space = url.indexOf(QChar::fromLatin1(' ')); space != -1) url = url.left(space); // Strip off the branch information to get just the repo if (url == QString::fromUtf8("Unknown")) url = QString::fromUtf8("https://github.com/FreeCAD/FreeCAD"); // Just take a guess // This may only create valid URLs for Github, but some other hosts use the same format so give it a shot... auto https = url.replace(QString::fromUtf8("git://"), QString::fromUtf8("https://")); https.replace(QString::fromUtf8(".git"), QString::fromUtf8("")); ui->labelBuildHash->setUrl(https + QString::fromUtf8("/commit/") + QString::fromStdString(it->second)); } } else { ui->labelHash->hide(); ui->labelBuildHash->hide(); } } void AboutDialog::showCredits() { auto creditsFileURL = QLatin1String(":/doc/CONTRIBUTORS"); QFile creditsFile(creditsFileURL); if (!creditsFile.open(QIODevice::ReadOnly | QIODevice::Text)) { return; } auto tab_credits = new QWidget(); tab_credits->setObjectName(QString::fromLatin1("tab_credits")); ui->tabWidget->addTab(tab_credits, tr("Credits")); auto hlayout = new QVBoxLayout(tab_credits); auto textField = new QTextBrowser(tab_credits); textField->setOpenExternalLinks(false); textField->setOpenLinks(false); hlayout->addWidget(textField); QString creditsHTML = QString::fromLatin1("

"); //: Header for bgbsww creditsHTML += tr("This version of FreeCAD is dedicated to the memory of Brad McLean, aka bgbsww."); //: Header for the Credits tab of the About screen creditsHTML += QString::fromLatin1("

"); creditsHTML += tr("Credits"); creditsHTML += QString::fromLatin1("

"); creditsHTML += tr("FreeCAD would not be possible without the contributions of"); creditsHTML += QString::fromLatin1(":

"); //: Header for the list of individual people in the Credits list. creditsHTML += tr("Individuals"); creditsHTML += QString::fromLatin1("

    "); QTextStream stream(&creditsFile); #if QT_VERSION < QT_VERSION_CHECK(6,0,0) stream.setCodec("UTF-8"); #endif QString line; while (stream.readLineInto(&line)) { if (!line.isEmpty()) { if (line == QString::fromLatin1("Firms")) { creditsHTML += QString::fromLatin1("

"); //: Header for the list of companies/organizations in the Credits list. creditsHTML += tr("Organizations"); creditsHTML += QString::fromLatin1("

    "); } else { creditsHTML += QString::fromLatin1("
  • ") + line + QString::fromLatin1("
  • "); } } } creditsHTML += QString::fromLatin1("
"); textField->setHtml(creditsHTML); } void AboutDialog::showLicenseInformation() { QString licenseFileURL = QString::fromLatin1("%1/LICENSE.html") .arg(QString::fromUtf8(App::Application::getHelpDir().c_str())); QFile licenseFile(licenseFileURL); if (licenseFile.open(QIODevice::ReadOnly | QIODevice::Text)) { QString licenseHTML = QString::fromUtf8(licenseFile.readAll()); const auto placeholder = QString::fromUtf8(""); licenseHTML.replace(placeholder, getAdditionalLicenseInformation()); ui->tabWidget->removeTab(1); // Hide the license placeholder widget auto tab_license = new QWidget(); tab_license->setObjectName(QString::fromLatin1("tab_license")); ui->tabWidget->addTab(tab_license, tr("License")); auto hlayout = new QVBoxLayout(tab_license); auto textField = new QTextBrowser(tab_license); textField->setOpenExternalLinks(true); textField->setOpenLinks(true); hlayout->addWidget(textField); textField->setHtml(licenseHTML); } else { QString info(QLatin1String("SUCH DAMAGES.
")); info += getAdditionalLicenseInformation(); QString lictext = ui->textBrowserLicense->toHtml(); lictext.replace(QString::fromLatin1("SUCH DAMAGES.
"), info); ui->textBrowserLicense->setHtml(lictext); } } QString AboutDialog::getAdditionalLicenseInformation() const { // Any additional piece of text to be added after the main license text goes below. // Please set title in

tags, license text in

tags // and add an


tag at the end to nicely separate license blocks QString info; #ifdef _USE_3DCONNEXION_SDK info += QString::fromUtf8( "

3D Mouse Support

" "

Development tools and related technology provided under license from 3Dconnexion.
" "Copyright © 1992–2012 3Dconnexion. All rights reserved.

" "
" ); #endif return info; } void AboutDialog::showLibraryInformation() { auto tab_library = new QWidget(); tab_library->setObjectName(QString::fromLatin1("tab_library")); ui->tabWidget->addTab(tab_library, tr("Libraries")); auto hlayout = new QVBoxLayout(tab_library); auto textField = new QTextBrowser(tab_library); textField->setOpenExternalLinks(true); hlayout->addWidget(textField); QString baseurl = QString::fromLatin1("file:///%1/ThirdPartyLibraries.html") .arg(QString::fromUtf8(App::Application::getHelpDir().c_str())); QUrl librariesFileUrl = QUrl(baseurl); textField->setSource(librariesFileUrl); } void AboutDialog::showCollectionInformation() { QString doc = QString::fromUtf8(App::Application::getHelpDir().c_str()); QString path = doc + QLatin1String("Collection.html"); if (!QFile::exists(path)) return; auto tab_collection = new QWidget(); tab_collection->setObjectName(QString::fromLatin1("tab_collection")); ui->tabWidget->addTab(tab_collection, tr("Collection")); auto hlayout = new QVBoxLayout(tab_collection); auto textField = new QTextBrowser(tab_collection); textField->setOpenExternalLinks(true); hlayout->addWidget(textField); textField->setSource(path); } void AboutDialog::showPrivacyPolicy() { auto policyFileURL = QLatin1String(":/doc/PRIVACY_POLICY"); QFile policyFile(policyFileURL); if (!policyFile.open(QIODevice::ReadOnly | QIODevice::Text)) { return; } auto text = QString::fromUtf8(policyFile.readAll()); auto tabPrivacyPolicy = new QWidget(); tabPrivacyPolicy->setObjectName(QString::fromLatin1("tabPrivacyPolicy")); ui->tabWidget->addTab(tabPrivacyPolicy, tr("Privacy Policy")); auto hLayout = new QVBoxLayout(tabPrivacyPolicy); auto textField = new QTextBrowser(tabPrivacyPolicy); textField->setOpenExternalLinks(true); hLayout->addWidget(textField); #if QT_VERSION < QT_VERSION_CHECK(5,15,0) // We can't actually render the markdown, so just display it as text textField->setText(text); #else textField->setMarkdown(text); #endif } void AboutDialog::linkActivated(const QUrl& link) { auto licenseView = new LicenseView(); licenseView->setAttribute(Qt::WA_DeleteOnClose); licenseView->show(); QString title = tr("License"); QString fragment = link.fragment(); if (fragment.startsWith(QLatin1String("_Toc"))) { QString prefix = fragment.mid(4); title = QString::fromLatin1("%1 %2").arg(prefix, title); } licenseView->setWindowTitle(title); getMainWindow()->addWindow(licenseView); licenseView->setSource(link); } void AboutDialog::copyToClipboard() { QString data; QTextStream str(&data); std::map& config = App::Application::Config(); std::map::iterator it; QString exe = QString::fromStdString(App::Application::getExecutableName()); QString major = QString::fromLatin1(config["BuildVersionMajor"].c_str()); QString minor = QString::fromLatin1(config["BuildVersionMinor"].c_str()); QString point = QString::fromLatin1(config["BuildVersionPoint"].c_str()); QString suffix = QString::fromLatin1(config["BuildVersionSuffix"].c_str()); QString build = QString::fromLatin1(config["BuildRevision"].c_str()); QString deskEnv = QProcessEnvironment::systemEnvironment().value(QStringLiteral("XDG_CURRENT_DESKTOP"), QString()); QString deskSess = QProcessEnvironment::systemEnvironment().value(QStringLiteral("DESKTOP_SESSION"), QString()); QStringList deskInfoList; QString deskInfo; if (!deskEnv.isEmpty()) { deskInfoList.append(deskEnv); } if (!deskSess.isEmpty()) { deskInfoList.append(deskSess); } if (qGuiApp->platformName() != QLatin1String("windows") && qGuiApp->platformName() != QLatin1String("cocoa")) { deskInfoList.append(qGuiApp->platformName()); } if(!deskInfoList.isEmpty()) { deskInfo = QLatin1String(" (") + deskInfoList.join(QLatin1String("/")) + QLatin1String(")"); } str << "OS: " << prettyProductInfoWrapper() << deskInfo << '\n'; if (QSysInfo::buildCpuArchitecture() == QSysInfo::currentCpuArchitecture()){ str << "Architecture: " << QSysInfo::buildCpuArchitecture() << "\n"; } else { str << "Architecture: " << QSysInfo::buildCpuArchitecture() << "(running on: " << QSysInfo::currentCpuArchitecture() << ")\n"; } str << "Version: " << major << "." << minor << "." << point << suffix << "." << build; char *appimage = getenv("APPIMAGE"); if (appimage) str << " AppImage"; char* snap = getenv("SNAP_REVISION"); if (snap) str << " Snap " << snap; str << '\n'; #if defined(_DEBUG) || defined(DEBUG) str << "Build type: Debug\n"; #elif defined(NDEBUG) str << "Build type: Release\n"; #elif defined(CMAKE_BUILD_TYPE) str << "Build type: " << CMAKE_BUILD_TYPE << '\n'; #else str << "Build type: Unknown\n"; #endif it = config.find("BuildRevisionBranch"); if (it != config.end()) str << "Branch: " << QString::fromUtf8(it->second.c_str()) << '\n'; it = config.find("BuildRevisionHash"); if (it != config.end()) str << "Hash: " << it->second.c_str() << '\n'; // report also the version numbers of the most important libraries in FreeCAD str << "Python " << PY_VERSION << ", "; str << "Qt " << QT_VERSION_STR << ", "; str << "Coin " << COIN_VERSION << ", "; str << "Vtk " << fcVtkVersion << ", "; #if defined(HAVE_OCC_VERSION) str << "OCC " << OCC_VERSION_MAJOR << "." << OCC_VERSION_MINOR << "." << OCC_VERSION_MAINTENANCE #ifdef OCC_VERSION_DEVELOPMENT << "." OCC_VERSION_DEVELOPMENT #endif << '\n'; #endif QLocale loc; str << "Locale: " << QLocale::languageToString(loc.language()) << "/" #if QT_VERSION < QT_VERSION_CHECK(6,6,0) << QLocale::countryToString(loc.country()) #else << QLocale::territoryToString(loc.territory()) #endif << " (" << loc.name() << ")"; if (loc != QLocale::system()) { loc = QLocale::system(); str << " [ OS: " << QLocale::languageToString(loc.language()) << "/" #if QT_VERSION < QT_VERSION_CHECK(6,6,0) << QLocale::countryToString(loc.country()) #else << QLocale::territoryToString(loc.territory()) #endif << " (" << loc.name() << ") ]"; } str << "\n"; // Add Stylesheet/Theme/Qtstyle information std::string styleSheet = App::GetApplication().GetParameterGroupByPath("User parameter:BaseApp/Preferences/MainWindow")->GetASCII("StyleSheet"); std::string theme = App::GetApplication().GetParameterGroupByPath("User parameter:BaseApp/Preferences/MainWindow")->GetASCII("Theme"); #if QT_VERSION >= QT_VERSION_CHECK(6, 1, 0) std::string style = qApp->style()->name().toStdString(); #else std::string style = App::GetApplication().GetParameterGroupByPath("User parameter:BaseApp/Preferences/MainWindow")->GetASCII("QtStyle"); if(style.empty()) { style = "Qt default"; } #endif if(styleSheet.empty()) { styleSheet = "unset"; } if(theme.empty()) { theme = "unset"; } str << "Stylesheet/Theme/QtStyle: " << QString::fromStdString(styleSheet) << "/" << QString::fromStdString(theme) << "/" << QString::fromStdString(style) << "\n"; // Add installed module information: auto modDir = fs::path(App::Application::getUserAppDataDir()) / "Mod"; bool firstMod = true; if (fs::exists(modDir) && fs::is_directory(modDir)) { for (const auto& mod : fs::directory_iterator(modDir)) { auto dirName = mod.path().filename().string(); if (dirName[0] == '.') // Ignore dot directories continue; if (firstMod) { firstMod = false; str << "Installed mods: \n"; } str << " * " << QString::fromStdString(mod.path().filename().string()); auto metadataFile = mod.path() / "package.xml"; if (fs::exists(metadataFile)) { App::Metadata metadata(metadataFile); if (metadata.version() != App::Meta::Version()) str << QLatin1String(" ") + QString::fromStdString(metadata.version().str()); } auto disablingFile = mod.path() / "ADDON_DISABLED"; if (fs::exists(disablingFile)) str << " (Disabled)"; str << "\n"; } } QClipboard* cb = QApplication::clipboard(); cb->setText(data); } // ---------------------------------------------------------------------------- /* TRANSLATOR Gui::LicenseView */ LicenseView::LicenseView(QWidget* parent) : MDIView(nullptr,parent,Qt::WindowFlags()) { browser = new QTextBrowser(this); browser->setOpenExternalLinks(true); browser->setOpenLinks(true); setCentralWidget(browser); } LicenseView::~LicenseView() = default; void LicenseView::setSource(const QUrl& url) { browser->setSource(url); } #include "moc_Splashscreen.cpp"