mirror of
https://github.com/86Box/86Box.git
synced 2026-02-23 18:08:20 -07:00
219 lines
5.5 KiB
C++
219 lines
5.5 KiB
C++
/*
|
|
* 86Box A hypervisor and IBM PC system emulator that specializes in
|
|
* running old operating systems and software designed for IBM
|
|
* PC systems and compatibles from 1981 through fairly recent
|
|
* system designs based on the PCI bus.
|
|
*
|
|
* This file is part of the 86Box distribution.
|
|
*
|
|
* Utility functions.
|
|
*
|
|
*
|
|
*
|
|
* Authors: Teemu Korhonen
|
|
*
|
|
* Copyright 2022 Teemu Korhonen
|
|
*/
|
|
#include <QDir>
|
|
#include <QFileInfo>
|
|
#include <QStringBuilder>
|
|
#include <QStringList>
|
|
#include <QWidget>
|
|
#include <QApplication>
|
|
#if QT_VERSION <= QT_VERSION_CHECK(5, 14, 0)
|
|
# include <QDesktopWidget>
|
|
#endif
|
|
#include <QUuid>
|
|
#include "qt_util.hpp"
|
|
|
|
#ifdef Q_OS_WINDOWS
|
|
# include <windows.h>
|
|
# include <dwmapi.h>
|
|
# ifndef DWMWA_WINDOW_CORNER_PREFERENCE
|
|
# define DWMWA_WINDOW_CORNER_PREFERENCE 33
|
|
# endif
|
|
# ifndef DWMWCP_DEFAULT
|
|
# define DWMWCP_DEFAULT 0
|
|
# endif
|
|
# ifndef DWMWCP_DONOTROUND
|
|
# define DWMWCP_DONOTROUND 1
|
|
# endif
|
|
#endif
|
|
|
|
extern "C" {
|
|
#include <86box/86box.h>
|
|
#include <86box/config.h>
|
|
#include <86box/device.h>
|
|
#include <86box/ini.h>
|
|
#include <86box/random.h>
|
|
#include <86box/thread.h>
|
|
#include <86box/timer.h>
|
|
#include <86box/network.h>
|
|
}
|
|
|
|
namespace util {
|
|
QScreen *
|
|
screenOfWidget(QWidget *widget)
|
|
{
|
|
#if QT_VERSION <= QT_VERSION_CHECK(5, 14, 0)
|
|
return QApplication::screens()[QApplication::desktop()->screenNumber(widget) == -1 ? 0 : QApplication::desktop()->screenNumber(widget)];
|
|
#else
|
|
return widget->screen();
|
|
#endif
|
|
}
|
|
|
|
#ifdef Q_OS_WINDOWS
|
|
|
|
bool
|
|
isWindowsLightTheme(void) {
|
|
if (color_scheme != 0) {
|
|
return (color_scheme == 1);
|
|
}
|
|
|
|
// based on https://stackoverflow.com/questions/51334674/how-to-detect-windows-10-light-dark-mode-in-win32-application
|
|
|
|
// The value is expected to be a REG_DWORD, which is a signed 32-bit little-endian
|
|
auto buffer = std::vector<char>(4);
|
|
auto cbData = static_cast<DWORD>(buffer.size() * sizeof(char));
|
|
auto res = RegGetValueW(
|
|
HKEY_CURRENT_USER,
|
|
L"Software\\Microsoft\\Windows\\CurrentVersion\\Themes\\Personalize",
|
|
L"AppsUseLightTheme",
|
|
RRF_RT_REG_DWORD, // expected value type
|
|
nullptr,
|
|
buffer.data(),
|
|
&cbData);
|
|
|
|
if (res != ERROR_SUCCESS) {
|
|
return 1;
|
|
}
|
|
|
|
// convert bytes written to our buffer to an int, assuming little-endian
|
|
auto i = int(buffer[3] << 24 |
|
|
buffer[2] << 16 |
|
|
buffer[1] << 8 |
|
|
buffer[0]);
|
|
|
|
return i == 1;
|
|
}
|
|
|
|
void
|
|
setWin11RoundedCorners(WId hwnd, bool enable)
|
|
{
|
|
auto cornerPreference = (enable ? DWMWCP_DEFAULT : DWMWCP_DONOTROUND);
|
|
DwmSetWindowAttribute((HWND) hwnd, DWMWA_WINDOW_CORNER_PREFERENCE, (LPCVOID) &cornerPreference, sizeof(cornerPreference));
|
|
}
|
|
#endif
|
|
|
|
QString
|
|
DlgFilter(std::initializer_list<QString> extensions, bool last)
|
|
{
|
|
QStringList temp;
|
|
|
|
for (auto ext : extensions) {
|
|
#ifdef Q_OS_UNIX
|
|
if (ext == "*") {
|
|
temp.append("*");
|
|
continue;
|
|
}
|
|
temp.append("*." % ext.toUpper());
|
|
#endif
|
|
temp.append("*." % ext);
|
|
}
|
|
|
|
#ifdef Q_OS_UNIX
|
|
temp.removeDuplicates();
|
|
#endif
|
|
return " (" % temp.join(' ') % ")" % (!last ? ";;" : "");
|
|
}
|
|
|
|
QString
|
|
DlgFilter(QStringList extensions, bool last)
|
|
{
|
|
QStringList temp;
|
|
|
|
for (auto ext : extensions) {
|
|
#ifdef Q_OS_UNIX
|
|
if (ext == "*") {
|
|
temp.append("*");
|
|
continue;
|
|
}
|
|
temp.append("*." % ext.toUpper());
|
|
#endif
|
|
temp.append("*." % ext);
|
|
}
|
|
|
|
#ifdef Q_OS_UNIX
|
|
temp.removeDuplicates();
|
|
#endif
|
|
return " (" % temp.join(' ') % ")" % (!last ? ";;" : "");
|
|
}
|
|
|
|
|
|
QString currentUuid()
|
|
{
|
|
return generateUuid(QString(cfg_path));
|
|
}
|
|
|
|
QString generateUuid(const QString &path)
|
|
{
|
|
auto dirPath = QFileInfo(path).dir().canonicalPath();
|
|
if(!dirPath.endsWith("/")) {
|
|
dirPath.append("/");
|
|
}
|
|
return QUuid::createUuidV5(QUuid{}, dirPath).toString(QUuid::WithoutBraces);
|
|
}
|
|
|
|
bool compareUuid()
|
|
{
|
|
// A uuid not set in the config file will have a zero length.
|
|
// Any uuid that is lower than the minimum length will be considered invalid
|
|
// and a new one will be generated
|
|
if (const auto currentUuidLength = QString(uuid).length(); currentUuidLength < UUID_MIN_LENGTH) {
|
|
storeCurrentUuid();
|
|
return true;
|
|
}
|
|
// Do not prompt on mismatch if the system does not have any configured NICs. Just update the uuid
|
|
if(!hasConfiguredNICs() && uuid != currentUuid()) {
|
|
storeCurrentUuid();
|
|
return true;
|
|
}
|
|
// The uuid appears to be a valid, at least by length.
|
|
// Compare with a simple string match
|
|
return uuid == currentUuid();
|
|
}
|
|
|
|
void
|
|
storeCurrentUuid()
|
|
{
|
|
strncpy(uuid, currentUuid().toUtf8().constData(), sizeof(uuid) - 1);
|
|
}
|
|
|
|
void
|
|
generateNewMacAdresses()
|
|
{
|
|
for (int i = 0; i < NET_CARD_MAX; ++i) {
|
|
auto net_card = net_cards_conf[i];
|
|
if (net_card.device_num != 0) {
|
|
const auto network_device = network_card_getdevice(net_card.device_num);
|
|
device_context_t device_context;
|
|
device_set_context(&device_context, network_device, i + 1);
|
|
auto generatedMac = QString::asprintf("%02X:%02X:%02X", random_generate(), random_generate(), random_generate()).toLower();
|
|
config_set_string(device_context.name, "mac", generatedMac.toUtf8().constData());
|
|
}
|
|
}
|
|
}
|
|
|
|
bool
|
|
hasConfiguredNICs()
|
|
{
|
|
for (int i = 0; i < NET_CARD_MAX; ++i) {
|
|
if (const auto net_card = net_cards_conf[i]; net_card.device_num != 0) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
}
|