2022-04-11 21:59:37 -04:00
|
|
|
#include "mainwindow.h"
|
|
|
|
|
2022-04-12 16:19:06 -04:00
|
|
|
#include <QWindow>
|
|
|
|
#include <QVulkanInstance>
|
2022-04-12 12:19:46 -04:00
|
|
|
#include <QHBoxLayout>
|
2022-04-11 21:59:37 -04:00
|
|
|
#include <QTableWidget>
|
|
|
|
#include <fmt/core.h>
|
|
|
|
#include <QListWidget>
|
2022-04-11 23:11:33 -04:00
|
|
|
#include <QVulkanWindow>
|
2022-04-12 00:30:17 -04:00
|
|
|
#include <QLineEdit>
|
|
|
|
#include <QResizeEvent>
|
2022-04-12 12:39:33 -04:00
|
|
|
#include <QComboBox>
|
2022-04-12 16:19:06 -04:00
|
|
|
#include <QTimer>
|
2022-04-11 21:59:37 -04:00
|
|
|
|
|
|
|
#include "gamedata.h"
|
|
|
|
#include "exhparser.h"
|
|
|
|
#include "exdparser.h"
|
|
|
|
#include "mdlparser.h"
|
|
|
|
|
2022-04-12 16:19:06 -04:00
|
|
|
#ifndef USE_STANDALONE_WINDOW
|
2022-04-11 23:11:33 -04:00
|
|
|
class VulkanWindow : public QWindow
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
VulkanWindow(Renderer* renderer, QVulkanInstance* instance) : m_renderer(renderer), m_instance(instance) {
|
|
|
|
setSurfaceType(VulkanSurface);
|
|
|
|
setVulkanInstance(instance);
|
|
|
|
}
|
|
|
|
|
|
|
|
void exposeEvent(QExposeEvent *) {
|
|
|
|
if (isExposed()) {
|
|
|
|
if (!m_initialized) {
|
|
|
|
m_initialized = true;
|
2022-04-12 00:30:17 -04:00
|
|
|
|
|
|
|
auto surface = m_instance->surfaceForWindow(this);
|
2022-04-12 12:19:46 -04:00
|
|
|
if(!m_renderer->initSwapchain(surface, width(), height()))
|
|
|
|
m_initialized = false;
|
|
|
|
else
|
|
|
|
render();
|
2022-04-11 23:11:33 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool event(QEvent *e) {
|
|
|
|
if (e->type() == QEvent::UpdateRequest)
|
|
|
|
render();
|
|
|
|
|
2022-04-12 00:30:17 -04:00
|
|
|
if (e->type() == QEvent::Resize) {
|
|
|
|
QResizeEvent* resizeEvent = (QResizeEvent*)e;
|
|
|
|
auto surface = m_instance->surfaceForWindow(this);
|
|
|
|
m_renderer->resize(surface, resizeEvent->size().width(), resizeEvent->size().height());
|
|
|
|
}
|
|
|
|
|
2022-04-11 23:11:33 -04:00
|
|
|
return QWindow::event(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
void render() {
|
2022-04-12 01:57:37 -04:00
|
|
|
m_renderer->render(models);
|
2022-04-12 00:30:17 -04:00
|
|
|
m_instance->presentQueued(this);
|
2022-04-11 23:11:33 -04:00
|
|
|
requestUpdate();
|
|
|
|
}
|
|
|
|
|
2022-04-12 01:57:37 -04:00
|
|
|
std::vector<RenderModel> models;
|
|
|
|
|
2022-04-11 23:11:33 -04:00
|
|
|
private:
|
|
|
|
bool m_initialized = false;
|
|
|
|
Renderer* m_renderer;
|
|
|
|
QVulkanInstance* m_instance;
|
|
|
|
};
|
2022-04-12 16:19:06 -04:00
|
|
|
#else
|
|
|
|
#include "standalonewindow.h"
|
|
|
|
#endif
|
2022-04-11 23:11:33 -04:00
|
|
|
|
2022-04-11 21:59:37 -04:00
|
|
|
MainWindow::MainWindow(GameData& data) : data(data) {
|
|
|
|
setWindowTitle("mdlviewer");
|
2022-04-12 12:19:46 -04:00
|
|
|
setMinimumSize(QSize(640, 480));
|
2022-04-11 21:59:37 -04:00
|
|
|
|
|
|
|
auto dummyWidget = new QWidget();
|
|
|
|
setCentralWidget(dummyWidget);
|
|
|
|
|
2022-04-12 12:19:46 -04:00
|
|
|
auto layout = new QHBoxLayout();
|
2022-04-11 21:59:37 -04:00
|
|
|
dummyWidget->setLayout(layout);
|
2022-04-11 23:11:33 -04:00
|
|
|
|
2022-04-12 12:19:46 -04:00
|
|
|
// smallclothes body
|
|
|
|
{
|
|
|
|
GearInfo info = {};
|
|
|
|
info.name = "Smallclothes Body";
|
|
|
|
info.slot = Slot::Body;
|
|
|
|
|
|
|
|
gears.push_back(info);
|
|
|
|
}
|
|
|
|
|
|
|
|
// smallclothes legs
|
|
|
|
{
|
|
|
|
GearInfo info = {};
|
|
|
|
info.name = "Smallclothes Legs";
|
|
|
|
info.slot = Slot::Legs;
|
|
|
|
|
|
|
|
gears.push_back(info);
|
|
|
|
}
|
|
|
|
|
2022-04-12 15:28:29 -04:00
|
|
|
auto exh = *data.readExcelSheet("Item");
|
|
|
|
|
|
|
|
auto path = getEXDFilename(exh, "item", getLanguageCode(Language::English), exh.pages[1]);
|
|
|
|
data.extractFile("exd/" + path, path);
|
|
|
|
auto exd = readEXD(exh, path, exh.pages[1]);
|
|
|
|
for(auto row : exd.rows) {
|
|
|
|
auto primaryModel = row.data[47].uint64Data;
|
|
|
|
auto secondaryModel = row.data[48].uint64Data;
|
|
|
|
|
|
|
|
int16_t parts[4];
|
|
|
|
memcpy(parts, &primaryModel, sizeof(int16_t) * 4);
|
|
|
|
|
|
|
|
GearInfo info = {};
|
|
|
|
info.name = row.data[9].data;
|
|
|
|
info.slot = Slot::Body;
|
|
|
|
info.modelInfo.primaryID = parts[0];
|
|
|
|
|
|
|
|
gears.push_back(info);
|
|
|
|
}
|
|
|
|
|
2022-04-12 12:19:46 -04:00
|
|
|
auto listWidget = new QListWidget();
|
|
|
|
for(auto gear : gears)
|
|
|
|
listWidget->addItem(gear.name.c_str());
|
|
|
|
|
|
|
|
listWidget->setMaximumWidth(200);
|
|
|
|
|
|
|
|
layout->addWidget(listWidget);
|
2022-04-11 23:11:33 -04:00
|
|
|
|
2022-04-12 00:30:17 -04:00
|
|
|
renderer = new Renderer();
|
2022-04-11 23:11:33 -04:00
|
|
|
|
2022-04-12 16:19:06 -04:00
|
|
|
auto viewportLayout = new QVBoxLayout();
|
|
|
|
layout->addLayout(viewportLayout);
|
|
|
|
|
|
|
|
#ifndef USE_STANDALONE_WINDOW
|
2022-04-12 12:39:33 -04:00
|
|
|
auto inst = new QVulkanInstance();
|
2022-04-12 00:30:17 -04:00
|
|
|
inst->setVkInstance(renderer->instance);
|
|
|
|
inst->setFlags(QVulkanInstance::Flag::NoDebugOutputRedirect);
|
|
|
|
inst->create();
|
2022-04-11 23:11:33 -04:00
|
|
|
|
2022-04-12 12:39:33 -04:00
|
|
|
vkWindow = new VulkanWindow(renderer, inst);
|
2022-04-12 00:30:17 -04:00
|
|
|
vkWindow->setVulkanInstance(inst);
|
2022-04-11 23:11:33 -04:00
|
|
|
|
|
|
|
auto widget = QWidget::createWindowContainer(vkWindow);
|
|
|
|
|
2022-04-12 12:39:33 -04:00
|
|
|
viewportLayout->addWidget(widget);
|
2022-04-12 16:19:06 -04:00
|
|
|
#else
|
|
|
|
standaloneWindow = new StandaloneWindow(renderer);
|
|
|
|
renderer->initSwapchain(standaloneWindow->getSurface(renderer->instance), 640, 480);
|
|
|
|
|
|
|
|
QTimer* timer = new QTimer();
|
|
|
|
connect(timer, &QTimer::timeout, this, [this] {
|
|
|
|
standaloneWindow->render();
|
|
|
|
});
|
|
|
|
timer->start(1000);
|
|
|
|
#endif
|
2022-04-12 12:19:46 -04:00
|
|
|
|
2022-04-12 12:39:33 -04:00
|
|
|
QComboBox* raceCombo = new QComboBox();
|
|
|
|
for(auto [race, raceName] : raceNames) {
|
|
|
|
raceCombo->addItem(raceName.data());
|
|
|
|
}
|
|
|
|
|
|
|
|
connect(raceCombo, qOverload<int>(&QComboBox::currentIndexChanged), [this](int index) {
|
|
|
|
currentRace = (Race)index;
|
|
|
|
refreshModel();
|
|
|
|
});
|
|
|
|
|
|
|
|
viewportLayout->addWidget(raceCombo);
|
|
|
|
|
|
|
|
connect(listWidget, &QListWidget::itemClicked, [this](QListWidgetItem* item) {
|
|
|
|
for(auto& gear : gears) {
|
|
|
|
if(gear.name == item->text().toStdString())
|
|
|
|
loadedGears = {&gear};
|
2022-04-12 12:19:46 -04:00
|
|
|
}
|
2022-04-12 12:39:33 -04:00
|
|
|
|
|
|
|
refreshModel();
|
2022-04-12 12:19:46 -04:00
|
|
|
});
|
2022-04-12 12:39:33 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::refreshModel() {
|
2022-04-12 16:19:06 -04:00
|
|
|
#ifdef USE_STANDALONE_WINDOW
|
|
|
|
standaloneWindow->models.clear();
|
|
|
|
#else
|
2022-04-12 12:39:33 -04:00
|
|
|
vkWindow->models.clear();
|
2022-04-12 16:19:06 -04:00
|
|
|
#endif
|
2022-04-12 12:39:33 -04:00
|
|
|
|
|
|
|
for(auto gear : loadedGears) {
|
2022-04-12 15:28:29 -04:00
|
|
|
QString modelID = QString("%1").arg(gear->modelInfo.primaryID, 4, 10, QLatin1Char('0'));
|
|
|
|
|
|
|
|
QString resolvedModelPath = QString("chara/equipment/e%1/model/c%2e%3_%4.mdl");
|
|
|
|
resolvedModelPath = resolvedModelPath.arg(modelID, raceIDs[currentRace].data(), modelID, slotToName[gear->slot].data());
|
2022-04-12 12:39:33 -04:00
|
|
|
|
|
|
|
data.extractFile(resolvedModelPath.toStdString(), "top.mdl");
|
2022-04-12 16:19:06 -04:00
|
|
|
|
|
|
|
#ifndef USE_STANDALONE_WINDOW
|
2022-04-12 12:39:33 -04:00
|
|
|
vkWindow->models.push_back(renderer->addModel(parseMDL("top.mdl")));
|
2022-04-12 16:19:06 -04:00
|
|
|
#else
|
|
|
|
standaloneWindow->models.push_back(renderer->addModel(parseMDL("top.mdl")));
|
|
|
|
#endif
|
2022-04-12 12:39:33 -04:00
|
|
|
}
|
2022-04-11 21:59:37 -04:00
|
|
|
}
|