#include "home_widget.h" #include "../../../client/settings/cache_settings.h" #include "../../../interface/widgets/tabs/tab_supervisor.h" #include "../../theme_manager.h" #include "../../window_main.h" #include "background_sources.h" #include "home_styled_button.h" #include "tutorial/tutorial_controller.h" #include #include #include #include #include #include #include #include HomeWidget::HomeWidget(QWidget *parent, TabSupervisor *_tabSupervisor) : QWidget(parent), tabSupervisor(_tabSupervisor), background("theme:backgrounds/home"), overlay("theme:cockatrice") { layout = new QGridLayout(this); backgroundSourceCard = new CardInfoPictureArtCropWidget(this); gradientColors = extractDominantColors(background); layout->addWidget(createButtons(), 1, 1, Qt::AlignVCenter | Qt::AlignHCenter); layout->setRowStretch(0, 1); layout->setRowStretch(2, 1); layout->setColumnStretch(0, 1); layout->setColumnStretch(2, 1); setLayout(layout); cardChangeTimer = new QTimer(this); connect(cardChangeTimer, &QTimer::timeout, this, &HomeWidget::updateRandomCard); initializeBackgroundFromSource(); updateConnectButton(tabSupervisor->getClient()->getStatus()); connect(tabSupervisor->getClient(), &RemoteClient::statusChanged, this, &HomeWidget::updateConnectButton); connect(&SettingsCache::instance(), &SettingsCache::homeTabBackgroundSourceChanged, this, &HomeWidget::initializeBackgroundFromSource); connect(&SettingsCache::instance(), &SettingsCache::homeTabBackgroundShuffleFrequencyChanged, this, &HomeWidget::onBackgroundShuffleFrequencyChanged); auto mainWindow = QtUtils::findParentOfType(this); if (mainWindow) { tutorialController = new TutorialController(mainWindow); } else { tutorialController = new TutorialController(this); } auto sequence = TutorialSequence(); sequence.addStep({connectButton, "Connect to a server to play here!"}); auto vdeStep = TutorialStep{visualDeckEditorButton, "Create a new deck from cards in the database here!"}; vdeStep.requiresInteraction = true; vdeStep.allowClickThrough = true; vdeStep.validationHint = "Open the deck editor to try it out!"; vdeStep.validationTiming = ValidationTiming::OnSignal; vdeStep.autoAdvanceOnValid = true; vdeStep.validator = []() { return true; }; vdeStep.signalSource = visualDeckEditorButton; vdeStep.signalName = SIGNAL(clicked()); sequence.addStep(vdeStep); sequence.addStep({visualDeckStorageButton, "Browse the decks in your local collection."}); sequence.addStep({visualDatabaseDisplayButton, "View the card database here."}); sequence.addStep( {edhrecButton, "Browse EDHRec, an external service designed to provide card recommendations for decks."}); sequence.addStep({archidektButton, "Browse Archidekt, an external service that allows users to store " "decklists and import them to your local collection."}); sequence.addStep({replaybutton, "View replays of your past games here."}); sequence.addStep({exitButton, "Exit the application."}); tutorialController->addSequence(sequence); // Lambda is cleaner to read than overloading this connect(&SettingsCache::instance(), &SettingsCache::homeTabDisplayCardNameChanged, this, [this] { repaint(); }); connect(&SettingsCache::instance(), &SettingsCache::themeChanged, this, &HomeWidget::initializeBackgroundFromSource); connect(&SettingsCache::instance(), &SettingsCache::themeChanged, this, &HomeWidget::updateButtonsToBackgroundColor); } void HomeWidget::showEvent(QShowEvent *event) { QWidget::showEvent(event); if (!tutorialStarted) { tutorialStarted = true; // Start on next event loop iteration so everything is fully painted QTimer::singleShot(3, tutorialController, [this] { tutorialController->start(); }); } } void HomeWidget::initializeBackgroundFromSource() { if (CardDatabaseManager::getInstance()->getLoadStatus() != LoadStatus::Ok) { connect(CardDatabaseManager::getInstance(), &CardDatabase::cardDatabaseLoadingFinished, this, &HomeWidget::initializeBackgroundFromSource); return; } auto backgroundSourceType = BackgroundSources::fromId(SettingsCache::instance().getHomeTabBackgroundSource()); switch (backgroundSourceType) { case BackgroundSources::Theme: cardChangeTimer->stop(); background = QPixmap("theme:backgrounds/home"); backgroundSourceDeck = DeckList(); backgroundSourceCard->setCard(ExactCard()); updateButtonsToBackgroundColor(); update(); break; case BackgroundSources::RandomCardArt: backgroundSourceDeck = DeckList(); updateRandomCard(); onBackgroundShuffleFrequencyChanged(); break; case BackgroundSources::DeckFileArt: loadBackgroundSourceDeck(); updateRandomCard(); onBackgroundShuffleFrequencyChanged(); break; } } void HomeWidget::loadBackgroundSourceDeck() { std::optional deckOpt = DeckLoader::loadFromFile( SettingsCache::instance().getDeckPath() + "background.cod", DeckFileFormat::Cockatrice, false); backgroundSourceDeck = deckOpt.has_value() ? deckOpt.value().deckList : DeckList(); } void HomeWidget::setRandomCard(ExactCard &newCard) { static constexpr int ATTEMPTS = 10; for (int i = 0; i < ATTEMPTS; ++i) { ExactCard tmpCard = CardDatabaseManager::query()->getRandomCard(); if (tmpCard != backgroundSourceCard->getCard() && tmpCard.getCardPtr()->getProperty("layout") == "normal" && tmpCard.getPrinting().getSet() != nullptr) { newCard = tmpCard; return; } } qWarning() << "failed to set random card image after" << ATTEMPTS << "attempts"; } void HomeWidget::updateRandomCard() { auto backgroundSourceType = BackgroundSources::fromId(SettingsCache::instance().getHomeTabBackgroundSource()); ExactCard newCard; switch (backgroundSourceType) { case BackgroundSources::Theme: break; case BackgroundSources::RandomCardArt: setRandomCard(newCard); break; case BackgroundSources::DeckFileArt: QList cardRefs = backgroundSourceDeck.getCardRefList(); ExactCard oldCard = backgroundSourceCard->getCard(); if (!cardRefs.empty()) { if (cardRefs.size() == 1) { newCard = CardDatabaseManager::query()->getCard(cardRefs.first()); } else { // Keep picking until different do { int idx = QRandomGenerator::global()->bounded(cardRefs.size()); newCard = CardDatabaseManager::query()->getCard(cardRefs.at(idx)); } while (newCard == oldCard); } } else { do { newCard = CardDatabaseManager::query()->getRandomCard(); } while (newCard == oldCard); } break; } if (!newCard) { return; } connect(newCard.getCardPtr().data(), &CardInfo::pixmapUpdated, this, &HomeWidget::updateBackgroundProperties); backgroundSourceCard->setCard(newCard); background = backgroundSourceCard->getBackground(); } void HomeWidget::onBackgroundShuffleFrequencyChanged() { cardChangeTimer->stop(); if (SettingsCache::instance().getHomeTabBackgroundShuffleFrequency() > 0) { cardChangeTimer->start(SettingsCache::instance().getHomeTabBackgroundShuffleFrequency() * 1000); } } void HomeWidget::updateBackgroundProperties() { background = backgroundSourceCard->getBackground(); updateButtonsToBackgroundColor(); update(); // Triggers repaint } void HomeWidget::updateButtonsToBackgroundColor() { gradientColors = extractDominantColors(background); for (HomeStyledButton *button : findChildren()) { button->updateStylesheet(gradientColors); button->update(); } } QGroupBox *HomeWidget::createButtons() { QGroupBox *box = new QGroupBox(this); box->setStyleSheet(R"( QGroupBox { font-size: 20px; color: white; /* Title text color */ background: transparent; } QGroupBox::title { color: white; subcontrol-origin: margin; subcontrol-position: top center; /* or top left / right */ } )"); box->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding); QVBoxLayout *boxLayout = new QVBoxLayout; boxLayout->setAlignment(Qt::AlignHCenter); QLabel *logoLabel = new QLabel; logoLabel->setPixmap(overlay.scaledToWidth(200, Qt::SmoothTransformation)); logoLabel->setAlignment(Qt::AlignCenter); boxLayout->addWidget(logoLabel); boxLayout->addSpacing(25); connectButton = new HomeStyledButton("Connect/Play", gradientColors); boxLayout->addWidget(connectButton); visualDeckEditorButton = new HomeStyledButton(tr("Create New Deck"), gradientColors); connect(visualDeckEditorButton, &QPushButton::clicked, tabSupervisor, [this] { tabSupervisor->openDeckInNewTab(LoadedDeck()); }); boxLayout->addWidget(visualDeckEditorButton); visualDeckStorageButton = new HomeStyledButton(tr("Browse Decks"), gradientColors); connect(visualDeckStorageButton, &QPushButton::clicked, tabSupervisor, [this] { tabSupervisor->actTabVisualDeckStorage(true); }); boxLayout->addWidget(visualDeckStorageButton); visualDatabaseDisplayButton = new HomeStyledButton(tr("Browse Card Database"), gradientColors); connect(visualDatabaseDisplayButton, &QPushButton::clicked, tabSupervisor, &TabSupervisor::addVisualDatabaseDisplayTab); boxLayout->addWidget(visualDatabaseDisplayButton); edhrecButton = new HomeStyledButton(tr("Browse EDHRec"), gradientColors); connect(edhrecButton, &QPushButton::clicked, tabSupervisor, &TabSupervisor::addEdhrecMainTab); boxLayout->addWidget(edhrecButton); archidektButton = new HomeStyledButton(tr("Browse Archidekt"), gradientColors); connect(archidektButton, &QPushButton::clicked, tabSupervisor, &TabSupervisor::addArchidektTab); boxLayout->addWidget(archidektButton); replaybutton = new HomeStyledButton(tr("View Replays"), gradientColors); connect(replaybutton, &QPushButton::clicked, tabSupervisor, [this] { tabSupervisor->actTabReplays(true); }); boxLayout->addWidget(replaybutton); if (qobject_cast(tabSupervisor->parentWidget())) { exitButton = new HomeStyledButton(tr("Quit"), gradientColors); connect(exitButton, &QPushButton::clicked, qobject_cast(tabSupervisor->parentWidget()), &MainWindow::actExit); boxLayout->addWidget(exitButton); } box->setLayout(boxLayout); return box; } void HomeWidget::updateConnectButton(const ClientStatus status) { disconnect(connectButton, &QPushButton::clicked, nullptr, nullptr); switch (status) { case StatusConnecting: connectButton->setText(tr("Connecting...")); connectButton->setEnabled(false); break; case StatusDisconnected: connectButton->setText(tr("Connect")); connectButton->setEnabled(true); connect(connectButton, &QPushButton::clicked, qobject_cast(tabSupervisor->parentWidget()), &MainWindow::actConnect); break; case StatusLoggedIn: connectButton->setText(tr("Play")); connectButton->setEnabled(true); connect(connectButton, &QPushButton::clicked, tabSupervisor, &TabSupervisor::switchToFirstAvailableNetworkTab); break; default: break; } } QPair HomeWidget::extractDominantColors(const QPixmap &pixmap) { if (themeManager->isBuiltInTheme() && SettingsCache::instance().getHomeTabBackgroundSource() == BackgroundSources::toId(BackgroundSources::Theme)) { return QPair(QColor::fromRgb(20, 140, 60), QColor::fromRgb(120, 200, 80)); } // Step 1: Downscale image for performance QImage image = pixmap.toImage() .scaled(64, 64, Qt::KeepAspectRatio, Qt::SmoothTransformation) .convertToFormat(QImage::Format_RGB32); QMap colorCount; // Step 2: Count quantized colors for (int y = 0; y < image.height(); ++y) { const QRgb *scanLine = reinterpret_cast(image.scanLine(y)); for (int x = 0; x < image.width(); ++x) { QColor color = QColor::fromRgb(scanLine[x]); int r = color.red() & 0xF0; int g = color.green() & 0xF0; int b = color.blue() & 0xF0; QRgb quantized = qRgb(r, g, b); colorCount[quantized]++; } } // Step 3: Sort by frequency QVector> sortedColors; for (auto it = colorCount.constBegin(); it != colorCount.constEnd(); ++it) { sortedColors.append(qMakePair(it.key(), it.value())); } std::sort(sortedColors.begin(), sortedColors.end(), [](const QPair &a, const QPair &b) { return a.second > b.second; }); // Step 4: Pick top two distinct colors QColor first = QColor(sortedColors.value(0).first); QColor second = first; for (int i = 1; i < sortedColors.size(); ++i) { QColor candidate = QColor(sortedColors[i].first); if (candidate != first) { second = candidate; break; } } return QPair(first, second); } void HomeWidget::paintEvent(QPaintEvent *event) { QPainter painter(this); painter.setRenderHint(QPainter::Antialiasing); if (!background.isNull()) { QSize widgetSize = size() * devicePixelRatio(); QPixmap toDraw = background.scaled(widgetSize, Qt::KeepAspectRatioByExpanding, Qt::SmoothTransformation); // Draw scaled background centered QSize bgSize = toDraw.size(); QPoint topLeft((widgetSize.width() - bgSize.width()) / (devicePixelRatio() * 2), // undo scaling for painter (widgetSize.height() - bgSize.height()) / (devicePixelRatio() * 2)); painter.drawPixmap(topLeft, toDraw); } // Draw translucent black overlay with rounded corners QRectF overlayRect(5, 5, width() - 10, height() - 10); QPainterPath roundedRectPath; roundedRectPath.addRoundedRect(overlayRect, 20, 20); QColor semiTransparentBlack(0, 0, 0, static_cast(255 * 0.33)); painter.fillPath(roundedRectPath, semiTransparentBlack); // Card name overlay (bottom-right) QString cardName; ExactCard card = backgroundSourceCard->getCard(); if (card) { cardName = card.getCardPtr()->getName(); if (card.getPrinting().getSet() != nullptr) { cardName += " (" + card.getPrinting().getSet()->getCorrectedShortName() + ") " + card.getPrinting().getProperty("num"); } } if (!cardName.isEmpty() && SettingsCache::instance().getHomeTabDisplayCardName()) { QFont font = painter.font(); font.setPointSize(14); font.setBold(true); painter.setFont(font); QFontMetrics fm(font); constexpr int padding = 10; constexpr int margin = 15; QRect textRect = fm.boundingRect(cardName); QRect bgRect(width() - textRect.width() - padding * 2 - margin, height() - textRect.height() - padding * 2 - margin, textRect.width() + padding * 2, textRect.height() + padding * 2); // Background bubble painter.setPen(Qt::NoPen); painter.setBrush(QColor(0, 0, 0, 160)); painter.drawRoundedRect(bgRect, 8, 8); // Text painter.setPen(Qt::white); painter.drawText(bgRect.adjusted(padding, padding, -padding, -padding), Qt::AlignRight | Qt::AlignVCenter, cardName); } QWidget::paintEvent(event); }