// SPDX-FileCopyrightText: 2016 Citra Emulator Project
// SPDX-License-Identifier: GPL-2.0-or-later

#include <chrono>
#include <forward_list>
#include <optional>

#include <QDateTimeEdit>
#include <QFileDialog>
#include <QGraphicsItem>
#include <QLineEdit>
#include <QMessageBox>
#include "common/settings.h"
#include "core/core.h"
#include "core/hle/service/time/time_manager.h"
#include "ui_configure_system.h"
#include "yuzu/configuration/config.h"
#include "yuzu/configuration/configuration_shared.h"
#include "yuzu/configuration/configure_system.h"
#include "yuzu/configuration/shared_widget.h"

constexpr std::array<u32, 7> LOCALE_BLOCKLIST{
    // pzzefezrpnkzeidfej
    // thhsrnhutlohsternp
    // BHH4CG          U
    // Raa1AB          S
    //  nn9
    //  ts
    0b0100011100001100000, // Japan
    0b0000001101001100100, // Americas
    0b0100110100001000010, // Europe
    0b0100110100001000010, // Australia
    0b0000000000000000000, // China
    0b0100111100001000000, // Korea
    0b0100111100001000000, // Taiwan
};

static bool IsValidLocale(u32 region_index, u32 language_index) {
    if (region_index >= LOCALE_BLOCKLIST.size()) {
        return false;
    }
    return ((LOCALE_BLOCKLIST.at(region_index) >> language_index) & 1) == 0;
}

ConfigureSystem::ConfigureSystem(
    Core::System& system_, std::shared_ptr<std::forward_list<ConfigurationShared::Tab*>> group,
    ConfigurationShared::TranslationMap& translations_, QWidget* parent)
    : Tab(group, parent), ui{std::make_unique<Ui::ConfigureSystem>()}, system{system_},
      translations{translations_} {
    ui->setupUi(this);

    Setup();

    connect(rng_seed_checkbox, &QCheckBox::stateChanged, this, [this](int state) {
        rng_seed_edit->setEnabled(state == Qt::Checked);
        if (state != Qt::Checked) {
            rng_seed_edit->setText(QStringLiteral("00000000"));
        }
    });

    connect(custom_rtc_checkbox, &QCheckBox::stateChanged, this, [this](int state) {
        custom_rtc_edit->setEnabled(state == Qt::Checked);
        if (state != Qt::Checked) {
            custom_rtc_edit->setDateTime(QDateTime::currentDateTime());
        }
    });

    const auto locale_check = [this](int index) {
        const auto region_index = combo_region->currentIndex();
        const auto language_index = combo_language->currentIndex();
        const bool valid_locale = IsValidLocale(region_index, language_index);
        ui->label_warn_invalid_locale->setVisible(!valid_locale);
        if (!valid_locale) {
            ui->label_warn_invalid_locale->setText(
                tr("Warning: \"%1\" is not a valid language for region \"%2\"")
                    .arg(combo_language->currentText())
                    .arg(combo_region->currentText()));
        }
    };

    connect(combo_language, qOverload<int>(&QComboBox::currentIndexChanged), this, locale_check);
    connect(combo_region, qOverload<int>(&QComboBox::currentIndexChanged), this, locale_check);

    SetConfiguration();
}

ConfigureSystem::~ConfigureSystem() = default;

void ConfigureSystem::changeEvent(QEvent* event) {
    if (event->type() == QEvent::LanguageChange) {
        RetranslateUI();
    }

    QWidget::changeEvent(event);
}

void ConfigureSystem::RetranslateUI() {
    ui->retranslateUi(this);
}

void ConfigureSystem::Setup() {
    const bool runtime_lock = !system.IsPoweredOn();
    auto& core_layout = *ui->core_widget->layout();
    auto& system_layout = *ui->system_widget->layout();

    std::map<std::string, QWidget*> core_hold{};
    std::map<bool, std::map<std::string, QWidget*>> system_hold{};

    std::forward_list<Settings::BasicSetting*> settings;
    auto push = [&settings](std::forward_list<Settings::BasicSetting*>& list) {
        for (auto setting : list) {
            settings.push_front(setting);
        }
    };

    push(Settings::values.linkage.by_category[Settings::Category::Core]);
    push(Settings::values.linkage.by_category[Settings::Category::System]);

    for (auto setting : settings) {
        ConfigurationShared::Widget* widget = [=]() {
            if (setting->Id() == Settings::values.custom_rtc_enabled.Id()) {
                return new ConfigurationShared::Widget(
                    setting, translations, this, runtime_lock, apply_funcs,
                    ConfigurationShared::RequestType::DateTimeEdit, true, 1.0f,
                    &Settings::values.custom_rtc);
            } else if (setting->Id() == Settings::values.rng_seed_enabled.Id()) {
                return new ConfigurationShared::Widget(setting, translations, this, runtime_lock,
                                                       apply_funcs,
                                                       ConfigurationShared::RequestType::HexEdit,
                                                       true, 1.0f, &Settings::values.rng_seed);
            } else {
                return new ConfigurationShared::Widget(setting, translations, this, runtime_lock,

                                                       apply_funcs);
            }
        }();

        if (!widget->Valid()) {
            delete widget;
            continue;
        }

        if (setting->Id() == Settings::values.rng_seed_enabled.Id()) {
            rng_seed_checkbox = widget->checkbox;
            rng_seed_edit = widget->line_edit;

            if (!Settings::values.rng_seed_enabled.GetValue()) {
                rng_seed_edit->setEnabled(false);
            }
        } else if (setting->Id() == Settings::values.custom_rtc_enabled.Id()) {
            custom_rtc_checkbox = widget->checkbox;
            custom_rtc_edit = widget->date_time_edit;

            custom_rtc_edit->setEnabled(Settings::values.custom_rtc_enabled.GetValue());
        } else if (setting->Id() == Settings::values.region_index.Id()) {

            combo_region = widget->combobox;
        } else if (setting->Id() == Settings::values.language_index.Id()) {
            combo_language = widget->combobox;
        }

        switch (setting->Category()) {
        case Settings::Category::Core:
            core_hold[setting->GetLabel()] = widget;
            break;
        case Settings::Category::System:
            system_hold[setting->IsEnum()].insert(std::pair{setting->GetLabel(), widget});
            break;
        default:
            delete widget;
        }
    }
    for (const auto& [label, widget] : core_hold) {
        core_layout.addWidget(widget);
    }
    for (const auto& [label, widget] : system_hold[true]) {
        system_layout.addWidget(widget);
    }
    for (const auto& [label, widget] : system_hold[false]) {
        system_layout.addWidget(widget);
    }
}

void ConfigureSystem::SetConfiguration() {}

void ConfigureSystem::ApplyConfiguration() {
    const bool powered_on = system.IsPoweredOn();
    for (const auto& func : apply_funcs) {
        func(powered_on);
    }
}