add persets

This commit is contained in:
brige 2025-11-01 18:08:02 +08:00
parent 00b9093481
commit 9df23631c8
14 changed files with 1173 additions and 626 deletions

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,46 @@
#pragma once
#include <QString>
#include <QVector>
/**
* @brief Model information structure
*
* Contains all information for a single preset model, including name, display name,
* description, mesh file path, icon path, and enabled status.
*/
struct ModelInfo {
QString name; ///< Model name (unique identifier)
QString displayName; ///< Display name (for UI display)
QString description; ///< Model description
QString meshFile; ///< 3D mesh file path
QString icon; ///< Icon file path
bool enabled; ///< Whether this model is enabled
/**
* @brief Default constructor
* Model is enabled by default
*/
ModelInfo() : enabled(true) {}
/**
* @brief Constructor with parameters
*/
ModelInfo(const QString& name, const QString& displayName, const QString& description,
const QString& meshFile, const QString& icon, bool enabled = true)
: name(name), displayName(displayName), description(description),
meshFile(meshFile), icon(icon), enabled(enabled) {}
};
/**
* @brief Model category structure
*
* Contains information for a model category, such as ships, satellites, missiles, jammers, etc.
* Each category contains multiple specific models.
*/
struct ModelCategory {
QString name; ///< Category name (unique identifier)
QString displayName; ///< Display name (for UI display)
QString icon; ///< Category icon path
QVector<ModelInfo> models; ///< All models in this category
};

View File

@ -102,6 +102,7 @@ bool PresetModelConfigParser::ParseXmlFile(const QString& filePath)
const char* modelDisplayName = modelElement->Attribute("displayName"); const char* modelDisplayName = modelElement->Attribute("displayName");
const char* description = modelElement->Attribute("description"); const char* description = modelElement->Attribute("description");
const char* meshFile = modelElement->Attribute("meshFile"); const char* meshFile = modelElement->Attribute("meshFile");
const char* modelIcon = modelElement->Attribute("icon");
const char* enabled = modelElement->Attribute("enabled"); const char* enabled = modelElement->Attribute("enabled");
if (!modelName) { if (!modelName) {
@ -114,6 +115,7 @@ bool PresetModelConfigParser::ParseXmlFile(const QString& filePath)
model.displayName = modelDisplayName ? QString::fromUtf8(modelDisplayName) : model.name; model.displayName = modelDisplayName ? QString::fromUtf8(modelDisplayName) : model.name;
model.description = description ? QString::fromUtf8(description) : ""; model.description = description ? QString::fromUtf8(description) : "";
model.meshFile = meshFile ? QString::fromUtf8(meshFile) : ""; model.meshFile = meshFile ? QString::fromUtf8(meshFile) : "";
model.icon = modelIcon ? QString::fromUtf8(modelIcon) : "";
model.enabled = enabled ? (strcmp(enabled, "true") == 0) : true; model.enabled = enabled ? (strcmp(enabled, "true") == 0) : true;
category.models.append(model); category.models.append(model);

View File

@ -3,24 +3,7 @@
#include <QString> #include <QString>
#include <QStringList> #include <QStringList>
#include <QMap> #include <QMap>
#include <QVector> #include "ModelInfo.h"
struct ModelInfo {
QString name;
QString displayName;
QString description;
QString meshFile;
bool enabled;
ModelInfo() : enabled(true) {}
};
struct ModelCategory {
QString name;
QString displayName;
QString icon;
QVector<ModelInfo> models;
};
class PresetModelConfigParser class PresetModelConfigParser
{ {

View File

@ -0,0 +1,144 @@
#include "ui/ModelBrowser/PresetModelListWidget.h"
#include <QListWidgetItem>
#include <QDrag>
#include <QMimeData>
#include <QPainter>
#include "common/RecourceHelper.h"
#include "common/SpdLogger.h"
PresetModelListWidget::PresetModelListWidget(QWidget* parent)
: QListWidget(parent)
{
setDragEnabled(true);
setDragDropMode(QAbstractItemView::DragOnly);
// Configure list widget for icon-text vertical layout
setViewMode(QListView::IconMode);
setResizeMode(QListView::Adjust);
setMovement(QListView::Static);
setGridSize(QSize(100, 80)); // Width x Height for each item
setIconSize(QSize(48, 48)); // Icon size
setSpacing(5); // Spacing between items
}
void PresetModelListWidget::startDrag(Qt::DropActions supportedActions) {
QListWidgetItem* item = currentItem();
if (nullptr == item) {
return;
}
QString modelType = property("modelType").toString();
QString modelName = item->data(Qt::UserRole).toString();
if (modelName.isEmpty()) {
modelName = item->text(); // Fallback to item text
}
QMimeData* mimeData = new QMimeData;
QString dragData = QString("%1|%2").arg(modelType, modelName);
mimeData->setData("application/x-preset-model", dragData.toUtf8());
// Create drag object
QDrag* drag = new QDrag(this);
drag->setMimeData(mimeData);
// Create drag pixmap with icon on top and text below
const int iconSize = 32;
const int textHeight = 20;
const int totalWidth = 80;
const int totalHeight = iconSize + textHeight + 5; // 5px spacing between icon and text
QPixmap pixmap(totalWidth, totalHeight);
pixmap.fill(Qt::transparent);
QPainter painter(&pixmap);
painter.setRenderHint(QPainter::Antialiasing);
// Draw semi-transparent background
painter.setBrush(QColor(255, 255, 255, 80)); // White with 70% opacity
painter.setPen(QPen(QColor(200, 200, 200, 200), 1)); // Light gray border
painter.drawRoundedRect(2, 2, totalWidth - 4, totalHeight - 4, 4, 4);
// Draw icon at the top center
QPixmap iconPixmap = item->icon().pixmap(iconSize, iconSize);
if (!iconPixmap.isNull()) {
int iconX = (totalWidth - iconSize) / 2;
painter.drawPixmap(iconX, 0, iconPixmap);
} else {
// Draw a default placeholder icon
int iconX = (totalWidth - iconSize) / 2;
painter.setPen(QPen(Qt::gray, 2));
painter.setBrush(Qt::lightGray);
painter.drawRect(iconX, 0, iconSize, iconSize);
painter.setPen(Qt::black);
painter.drawText(iconX, 0, iconSize, iconSize, Qt::AlignCenter, "?");
}
// Draw text below the icon
painter.setPen(Qt::black);
QFont font = painter.font();
font.setPointSize(8);
painter.setFont(font);
QRect textRect(0, iconSize + 5, totalWidth, textHeight);
painter.drawText(textRect, Qt::AlignCenter | Qt::TextWordWrap, item->text());
drag->setPixmap(pixmap);
// Set hot spot to the center of the icon (not the entire pixmap)
int hotSpotX = totalWidth / 2;
int hotSpotY = iconSize / 2; // Center of the icon part only
drag->setHotSpot(QPoint(hotSpotX, hotSpotY));
// Execute drag
drag->exec(Qt::CopyAction);
}
void PresetModelListWidget::setModelType(const QString& modelType) {
modelType_ = modelType;
setProperty("modelType", modelType_);
}
QString PresetModelListWidget::getModelType() const {
return modelType_;
}
void PresetModelListWidget::setModelList(const QVector<ModelInfo>& models) {
modelList_ = models;
clear();
for (const ModelInfo& modelInfo : models) {
if (!modelInfo.enabled) continue; // Skip disabled models
QListWidgetItem* item = new QListWidgetItem(modelInfo.displayName.isEmpty() ? modelInfo.name : modelInfo.displayName);
item->setFlags(item->flags() | Qt::ItemIsDragEnabled);
// Use icon from ModelInfo if available
if (!modelInfo.icon.isEmpty()) {
QString presetsPath = QString("%1/%2").arg(RecourceHelper::Get().GetResourcesPath(), modelInfo.icon);
QIcon icon(presetsPath);
item->setIcon(icon);
} else {
// Create a default icon if no icon is specified
QPixmap pixmap(48, 48);
pixmap.fill(Qt::lightGray);
QPainter painter(&pixmap);
painter.setPen(Qt::black);
painter.drawRect(0, 0, 47, 47);
painter.drawText(pixmap.rect(), Qt::AlignCenter, "?");
item->setIcon(QIcon(pixmap));
}
// Store model name as simple string
item->setData(Qt::UserRole, modelInfo.name);
addItem(item);
}
}
QVector<ModelInfo> PresetModelListWidget::getModelList() const {
return modelList_;
}

View File

@ -0,0 +1,23 @@
#include <QListWidget>
#include <QString>
#include "ui/ModelBrowser/ModelInfo.h"
class PresetModelListWidget : public QListWidget
{
Q_OBJECT
public:
explicit PresetModelListWidget(QWidget* parent = nullptr);
void setModelType(const QString& modelType);
QString getModelType() const;
void setModelList(const QVector<ModelInfo>& models);
QVector<ModelInfo> getModelList() const;
protected:
void startDrag(Qt::DropActions supportedActions);
private:
QString modelType_;
QVector<ModelInfo> modelList_;
};

View File

@ -8,11 +8,14 @@
#include <QPixmap> #include <QPixmap>
#include <QPainter> #include <QPainter>
#include <QVBoxLayout> #include <QVBoxLayout>
#include <QMetaType>
#include <QHBoxLayout> #include <QHBoxLayout>
#include <QLabel> #include <QLabel>
#include <QIcon> #include <QIcon>
#include <QMouseEvent> #include <QMouseEvent>
#include <QFontMetrics>
#include "common/RecourceHelper.h"
#include "common/SpdLogger.h" #include "common/SpdLogger.h"
#include "ui/DockTitleBar.h" #include "ui/DockTitleBar.h"
#include "ui/DockWidget.h" #include "ui/DockWidget.h"
@ -23,6 +26,7 @@
PresetModelPanel::PresetModelPanel(QWidget *parent) PresetModelPanel::PresetModelPanel(QWidget *parent)
: QWidget(parent) : QWidget(parent)
, ui(new Ui::PresetModelPanel) , ui(new Ui::PresetModelPanel)
, dragSourceWidget_(nullptr)
{ {
ui->setupUi(this); ui->setupUi(this);
InitUI(); InitUI();
@ -50,13 +54,14 @@ void PresetModelPanel::AttachDock(DockWidget* dockWidget)
void PresetModelPanel::InitUI() void PresetModelPanel::InitUI()
{ {
// Try to load model data from configuration files // Try to load model data from configuration files
if (m_configParser.LoadAllConfigs("workspace/presets")) { QString presetsPath = QString("%1/presets").arg(RecourceHelper::Get().GetResourcesPath());
if (configParser_.LoadAllConfigs(presetsPath)) {
// Successfully loaded from configuration files // Successfully loaded from configuration files
LoadModelsFromConfig(); LoadModelsFromConfig();
} else { } else {
// Failed to load configuration files, use hardcoded defaults as fallback // Failed to load configuration files, use hardcoded defaults as fallback
LOG_WARN("Failed to load preset models from config files, using hardcoded defaults. Error: {}", LOG_WARN("Failed to load preset models from config files, using hardcoded defaults. Error: {}",
m_configParser.GetLastError().toStdString()); configParser_.GetLastError().toStdString());
LoadDefaultModels(); LoadDefaultModels();
} }
} }
@ -64,161 +69,175 @@ void PresetModelPanel::InitUI()
void PresetModelPanel::LoadModelsFromConfig() void PresetModelPanel::LoadModelsFromConfig()
{ {
// Load ship models // Load ship models
QStringList shipModels = m_configParser.GetModelNames("Ship"); ModelCategory shipCategory = configParser_.GetCategory("Ship");
if (!shipModels.isEmpty()) { if (!shipCategory.models.isEmpty()) {
PopulateModelList(ui->shipList, "Ship", shipModels); ui->shipList->setModelType("Ship");
ui->shipList->setModelList(shipCategory.models);
} }
// Load satellite models // Load satellite models
QStringList satelliteModels = m_configParser.GetModelNames("Satellite"); ModelCategory satelliteCategory = configParser_.GetCategory("Satellite");
if (!satelliteModels.isEmpty()) { if (!satelliteCategory.models.isEmpty()) {
PopulateModelList(ui->satelliteList, "Satellite", satelliteModels); ui->satelliteList->setModelType("Satellite");
ui->satelliteList->setModelList(satelliteCategory.models);
} }
// Load missile models // Load missile models
QStringList missileModels = m_configParser.GetModelNames("Missile"); ModelCategory missileCategory = configParser_.GetCategory("Missile");
if (!missileModels.isEmpty()) { if (!missileCategory.models.isEmpty()) {
PopulateModelList(ui->missileList, "Missile", missileModels); ui->missileList->setModelType("Missile");
ui->missileList->setModelList(missileCategory.models);
} }
// Load jammer models // Load jammer models
QStringList jammerModels = m_configParser.GetModelNames("Jammer"); ModelCategory jammerCategory = configParser_.GetCategory("Jammer");
if (!jammerModels.isEmpty()) { if (!jammerCategory.models.isEmpty()) {
PopulateModelList(ui->jammerList, "Jammer", jammerModels); ui->jammerList->setModelType("Jammer");
ui->jammerList->setModelList(jammerCategory.models);
} }
} }
void PresetModelPanel::LoadDefaultModels() void PresetModelPanel::LoadDefaultModels()
{ {
// Use hardcoded default models as fallback mechanism // Use hardcoded default models as fallback mechanism
QStringList shipModels = { QVector<ModelInfo> shipModels = {
"Destroyer", {"destroyer", "Destroyer", "Naval destroyer vessel", "", ":/icons/ship.png", true},
"Aircraft Carrier", {"aircraft_carrier", "Aircraft Carrier", "Large naval vessel", "", ":/icons/ship.png", true},
"Frigate", {"frigate", "Frigate", "Fast naval vessel", "", ":/icons/ship.png", true},
"Submarine" {"submarine", "Submarine", "Underwater vessel", "", ":/icons/ship.png", true}
}; };
PopulateModelList(ui->shipList, "Ship", shipModels); ui->shipList->setModelType("Ship");
ui->shipList->setModelList(shipModels);
QStringList satelliteModels = { QVector<ModelInfo> satelliteModels = {
"Communication Satellite", {"comm_satellite", "Communication Satellite", "Communication satellite", "", ":/icons/satellite.png", true},
"Weather Satellite", {"weather_satellite", "Weather Satellite", "Weather monitoring satellite", "", ":/icons/satellite.png", true},
"GPS Satellite", {"gps_satellite", "GPS Satellite", "GPS navigation satellite", "", ":/icons/satellite.png", true},
"Spy Satellite" {"spy_satellite", "Spy Satellite", "Reconnaissance satellite", "", ":/icons/satellite.png", true}
}; };
PopulateModelList(ui->satelliteList, "Satellite", satelliteModels); ui->satelliteList->setModelType("Satellite");
ui->satelliteList->setModelList(satelliteModels);
QStringList missileModels = { QVector<ModelInfo> missileModels = {
"Cruise Missile", {"cruise_missile", "Cruise Missile", "Long-range cruise missile", "", ":/icons/missile.png", true},
"Ballistic Missile", {"ballistic_missile", "Ballistic Missile", "Ballistic missile", "", ":/icons/missile.png", true},
"Anti-Ship Missile", {"anti_ship_missile", "Anti-Ship Missile", "Anti-ship missile", "", ":/icons/missile.png", true},
"Surface-to-Air Missile" {"sam_missile", "Surface-to-Air Missile", "Surface-to-air missile", "", ":/icons/missile.png", true}
}; };
PopulateModelList(ui->missileList, "Missile", missileModels); ui->missileList->setModelType("Missile");
ui->missileList->setModelList(missileModels);
QStringList jammerModels = { QVector<ModelInfo> jammerModels = {
"Electronic Jammer", {"electronic_jammer", "Electronic Jammer", "Electronic warfare jammer", "", ":/icons/jammer.png", true},
"Communication Jammer", {"comm_jammer", "Communication Jammer", "Communication jammer", "", ":/icons/jammer.png", true},
"Radar Jammer", {"radar_jammer", "Radar Jammer", "Radar jamming device", "", ":/icons/jammer.png", true},
"GPS Jammer" {"gps_jammer", "GPS Jammer", "GPS jamming device", "", ":/icons/jammer.png", true}
}; };
PopulateModelList(ui->jammerList, "Jammer", jammerModels); ui->jammerList->setModelType("Jammer");
ui->jammerList->setModelList(jammerModels);
} }
void PresetModelPanel::PopulateModelList(QListWidget* listWidget, const QString& modelType, const QStringList& models) // void PresetModelPanel::OnModelItemDoubleClicked(QListWidgetItem* item)
{ // {
if (!listWidget) return; // if (!item) return;
// Set the model type as a property for later use // // Get the list widget that contains this item
listWidget->setProperty("modelType", modelType); // QListWidget* listWidget = item->listWidget();
// if (!listWidget) return;
for (const QString& model : models) { // QString modelType = listWidget->property("modelType").toString();
QListWidgetItem* item = new QListWidgetItem(model);
item->setFlags(item->flags() | Qt::ItemIsDragEnabled);
if (modelType == "Ship") { // // Get model name from stored data
item->setIcon(QIcon(":/icons/ship.png")); // QString modelName = item->data(Qt::UserRole).toString();
} else if (modelType == "Satellite") { // if (modelName.isEmpty()) {
item->setIcon(QIcon(":/icons/satellite.png")); // modelName = item->text(); // Fallback to display text
} else if (modelType == "Missile") { // }
item->setIcon(QIcon(":/icons/missile.png"));
} else if (modelType == "Jammer") {
item->setIcon(QIcon(":/icons/jammer.png"));
}
listWidget->addItem(item); // LOG_INFO("Model double-clicked: {} of type {}", qPrintable(modelName), qPrintable(modelType));
}
connect(listWidget, &QListWidget::itemDoubleClicked, this, &PresetModelPanel::OnModelItemDoubleClicked); // // Emit signal for model selection
} // emit ModelDropped(modelType, modelName);
// }
void PresetModelPanel::SetupDragAndDrop(QListWidget* listWidget) // bool PresetModelPanel::eventFilter(QObject* obj, QEvent* event)
{ // {
listWidget->setDragDropMode(QAbstractItemView::DragOnly); // QListWidget* listWidget = qobject_cast<QListWidget*>(obj);
listWidget->setDefaultDropAction(Qt::CopyAction); // if (!listWidget) {
} // return QWidget::eventFilter(obj, event);
// }
void PresetModelPanel::OnModelItemDoubleClicked(QListWidgetItem* item) // if (event->type() == QEvent::MouseButtonPress) {
{ // QMouseEvent* mouseEvent = static_cast<QMouseEvent*>(event);
if (!item) return; // if (mouseEvent->button() == Qt::LeftButton) {
// dragStartPosition_ = mouseEvent->pos();
// dragSourceWidget_ = listWidget;
// }
// } else if (event->type() == QEvent::MouseMove) {
// QMouseEvent* mouseEvent = static_cast<QMouseEvent*>(event);
// if (!(mouseEvent->buttons() & Qt::LeftButton)) {
// return QWidget::eventFilter(obj, event);
// }
QListWidget* listWidget = item->listWidget(); // if ((mouseEvent->pos() - dragStartPosition_).manhattanLength() < QApplication::startDragDistance()) {
if (!listWidget) return; // return QWidget::eventFilter(obj, event);
// }
QString modelType = listWidget->property("modelType").toString(); // QListWidgetItem* item = listWidget->itemAt(dragStartPosition_);
QString modelName = item->text(); // if (!item) {
// return QWidget::eventFilter(obj, event);
// }
emit ModelDropped(modelType, modelName); // QString modelType = listWidget->property("modelType").toString();
}
// DraggableListWidget implementation // // Get model name from stored data
DraggableListWidget::DraggableListWidget(const QString& modelType, QWidget* parent) // QString modelName = item->data(Qt::UserRole).toString();
: QListWidget(parent), modelType_(modelType) // if (modelName.isEmpty()) {
{ // modelName = item->text(); // Fallback to item text
setProperty("modelType", modelType); // }
}
void DraggableListWidget::mousePressEvent(QMouseEvent* event) // // Create mime data
{ // QMimeData* mimeData = new QMimeData;
if (event->button() == Qt::LeftButton) { // QString dragData = QString("%1|%2").arg(modelType, modelName);
dragStartPosition_ = event->pos(); // mimeData->setData("application/x-preset-model", dragData.toUtf8());
}
QListWidget::mousePressEvent(event);
}
void DraggableListWidget::mouseMoveEvent(QMouseEvent* event) { // // Create drag object
if (!(event->buttons() & Qt::LeftButton)) { // QDrag* drag = new QDrag(this);
return; // drag->setMimeData(mimeData);
}
if ((event->pos() - dragStartPosition_).manhattanLength() < QApplication::startDragDistance()) { // // Set drag pixmap (icon or text)
return; // QPixmap pixmap(item->icon().pixmap(32, 32));
} // if (pixmap.isNull()) {
// pixmap = QPixmap(100, 30);
// pixmap.fill(Qt::transparent);
// QPainter painter(&pixmap);
// painter.setPen(Qt::black);
// painter.drawText(pixmap.rect(), Qt::AlignCenter, item->text());
// }
// drag->setPixmap(pixmap);
QListWidgetItem* item = itemAt(dragStartPosition_); // // Execute drag
if (!item) { // Qt::DropAction dropAction = drag->exec(Qt::CopyAction);
return; // Q_UNUSED(dropAction);
}
QDrag* drag = new QDrag(this); // return true;
QMimeData* mimeData = new QMimeData; // }
QString modelType = this->property("modelType").toString(); // return QWidget::eventFilter(obj, event);
QString modelName = item->text(); // }
QString dragData = QString("%1|%2").arg(modelType, modelName);
mimeData->setData("application/x-preset-model", dragData.toUtf8()); // // DraggableListWidget implementation
drag->setMimeData(mimeData); // DraggableListWidget::DraggableListWidget(const QString& modelType, QWidget* parent)
// : QListWidget(parent), modelType_(modelType)
// {
// setProperty("modelType", modelType);
// }
QPixmap pixmap(item->icon().pixmap(32, 32)); // void DraggableListWidget::mousePressEvent(QMouseEvent* event)
if (pixmap.isNull()) { // {
pixmap = QPixmap(100, 30); // QListWidget::mousePressEvent(event);
pixmap.fill(Qt::transparent); // }
QPainter painter(&pixmap);
painter.setPen(Qt::black);
painter.drawText(pixmap.rect(), Qt::AlignCenter, modelName);
}
drag->setPixmap(pixmap);
Qt::DropAction dropAction = drag->exec(Qt::CopyAction); // void DraggableListWidget::mouseMoveEvent(QMouseEvent* event)
Q_UNUSED(dropAction); // {
} // QListWidget::mouseMoveEvent(event);
// }

View File

@ -10,14 +10,18 @@
#include <QMimeData> #include <QMimeData>
#include <QDrag> #include <QDrag>
#include <QMouseEvent> #include <QMouseEvent>
#include <QEvent>
#include <QStyledItemDelegate>
#include <QPainter>
#include <QApplication>
#include "ModelInfo.h"
#include "PresetModelConfigParser.h" #include "PresetModelConfigParser.h"
QT_BEGIN_NAMESPACE
namespace Ui { class PresetModelPanel; } namespace Ui { class PresetModelPanel; }
QT_END_NAMESPACE
class DockWidget; class DockWidget;
/** /**
* @brief Preset model panel class * @brief Preset model panel class
* *
@ -40,42 +44,12 @@ public:
*/ */
void AttachDock(DockWidget* dockWidget); void AttachDock(DockWidget* dockWidget);
Q_SIGNALS:
/**
* @brief Signal emitted when model is dragged to scene
* @param modelType Model type
* @param modelName Model name
* @param position Scene position (optional)
*/
void ModelDropped(const QString& modelType, const QString& modelName, const QPointF& position = QPointF());
private slots:
/**
* @brief Handle model item double click event
* @param item Clicked item
*/
void OnModelItemDoubleClicked(QListWidgetItem* item);
private: private:
/** /**
* @brief Initialize user interface * @brief Initialize user interface
*/ */
void InitUI(); void InitUI();
/**
* @brief Populate model list with items
* @param listWidget List widget to populate
* @param modelType Model type
* @param models Model name list
*/
void PopulateModelList(QListWidget* listWidget, const QString& modelType, const QStringList& models);
/**
* @brief Setup drag and drop functionality
* @param listWidget List widget to setup drag and drop
*/
void SetupDragAndDrop(QListWidget* listWidget);
/** /**
* @brief Load models from configuration file * @brief Load models from configuration file
*/ */
@ -88,26 +62,9 @@ private:
private: private:
Ui::PresetModelPanel *ui; Ui::PresetModelPanel *ui;
PresetModelConfigParser m_configParser; PresetModelConfigParser configParser_;
};
/** // Drag and drop support
* @brief Custom list widget with drag and drop support
*/
class DraggableListWidget : public QListWidget {
Q_OBJECT
public:
explicit DraggableListWidget(const QString& modelType, QWidget* parent = nullptr);
Q_SIGNALS:
void ItemDragged(const QString& modelType, const QString& modelName);
protected:
void mousePressEvent(QMouseEvent* event) override;
void mouseMoveEvent(QMouseEvent* event) override;
private:
QString modelType_;
QPoint dragStartPosition_; QPoint dragStartPosition_;
QListWidget* dragSourceWidget_;
}; };

View File

@ -14,6 +14,9 @@
<string>Preset Models</string> <string>Preset Models</string>
</property> </property>
<layout class="QVBoxLayout" name="mainLayout"> <layout class="QVBoxLayout" name="mainLayout">
<property name="spacing">
<number>5</number>
</property>
<property name="leftMargin"> <property name="leftMargin">
<number>5</number> <number>5</number>
</property> </property>
@ -26,16 +29,10 @@
<property name="bottomMargin"> <property name="bottomMargin">
<number>5</number> <number>5</number>
</property> </property>
<property name="spacing">
<number>5</number>
</property>
<item> <item>
<widget class="QToolBox" name="toolBox"> <widget class="QToolBox" name="PresetModelToolBox">
<property name="objectName">
<string>PresetModelToolBox</string>
</property>
<property name="currentIndex"> <property name="currentIndex">
<number>0</number> <number>1</number>
</property> </property>
<widget class="QWidget" name="shipPage"> <widget class="QWidget" name="shipPage">
<property name="geometry"> <property name="geometry">
@ -43,15 +40,30 @@
<x>0</x> <x>0</x>
<y>0</y> <y>0</y>
<width>290</width> <width>290</width>
<height>540</height> <height>470</height>
</rect> </rect>
</property> </property>
<attribute name="label"> <attribute name="label">
<string>Ships</string> <string>Ships</string>
</attribute> </attribute>
<layout class="QVBoxLayout" name="shipLayout"> <layout class="QVBoxLayout" name="shipLayout">
<property name="spacing">
<number>0</number>
</property>
<property name="leftMargin">
<number>0</number>
</property>
<property name="topMargin">
<number>0</number>
</property>
<property name="rightMargin">
<number>0</number>
</property>
<property name="bottomMargin">
<number>0</number>
</property>
<item> <item>
<widget class="QListWidget" name="shipList"> <widget class="PresetModelListWidget" name="shipList">
<property name="dragDropMode"> <property name="dragDropMode">
<enum>QAbstractItemView::DragOnly</enum> <enum>QAbstractItemView::DragOnly</enum>
</property> </property>
@ -68,15 +80,30 @@
<x>0</x> <x>0</x>
<y>0</y> <y>0</y>
<width>290</width> <width>290</width>
<height>540</height> <height>470</height>
</rect> </rect>
</property> </property>
<attribute name="label"> <attribute name="label">
<string>Satellites</string> <string>Satellites</string>
</attribute> </attribute>
<layout class="QVBoxLayout" name="satelliteLayout"> <layout class="QVBoxLayout" name="satelliteLayout">
<property name="spacing">
<number>0</number>
</property>
<property name="leftMargin">
<number>0</number>
</property>
<property name="topMargin">
<number>0</number>
</property>
<property name="rightMargin">
<number>0</number>
</property>
<property name="bottomMargin">
<number>0</number>
</property>
<item> <item>
<widget class="QListWidget" name="satelliteList"> <widget class="PresetModelListWidget" name="satelliteList">
<property name="dragDropMode"> <property name="dragDropMode">
<enum>QAbstractItemView::DragOnly</enum> <enum>QAbstractItemView::DragOnly</enum>
</property> </property>
@ -93,15 +120,30 @@
<x>0</x> <x>0</x>
<y>0</y> <y>0</y>
<width>290</width> <width>290</width>
<height>540</height> <height>470</height>
</rect> </rect>
</property> </property>
<attribute name="label"> <attribute name="label">
<string>Missiles</string> <string>Missiles</string>
</attribute> </attribute>
<layout class="QVBoxLayout" name="missileLayout"> <layout class="QVBoxLayout" name="missileLayout">
<property name="spacing">
<number>0</number>
</property>
<property name="leftMargin">
<number>0</number>
</property>
<property name="topMargin">
<number>0</number>
</property>
<property name="rightMargin">
<number>0</number>
</property>
<property name="bottomMargin">
<number>0</number>
</property>
<item> <item>
<widget class="QListWidget" name="missileList"> <widget class="PresetModelListWidget" name="missileList">
<property name="dragDropMode"> <property name="dragDropMode">
<enum>QAbstractItemView::DragOnly</enum> <enum>QAbstractItemView::DragOnly</enum>
</property> </property>
@ -118,15 +160,30 @@
<x>0</x> <x>0</x>
<y>0</y> <y>0</y>
<width>290</width> <width>290</width>
<height>540</height> <height>470</height>
</rect> </rect>
</property> </property>
<attribute name="label"> <attribute name="label">
<string>Jammers</string> <string>Jammers</string>
</attribute> </attribute>
<layout class="QVBoxLayout" name="jammerLayout"> <layout class="QVBoxLayout" name="jammerLayout">
<property name="spacing">
<number>0</number>
</property>
<property name="leftMargin">
<number>0</number>
</property>
<property name="topMargin">
<number>0</number>
</property>
<property name="rightMargin">
<number>0</number>
</property>
<property name="bottomMargin">
<number>0</number>
</property>
<item> <item>
<widget class="QListWidget" name="jammerList"> <widget class="PresetModelListWidget" name="jammerList">
<property name="dragDropMode"> <property name="dragDropMode">
<enum>QAbstractItemView::DragOnly</enum> <enum>QAbstractItemView::DragOnly</enum>
</property> </property>
@ -141,6 +198,13 @@
</item> </item>
</layout> </layout>
</widget> </widget>
<customwidgets>
<customwidget>
<class>PresetModelListWidget</class>
<extends>QListWidget</extends>
<header>ui/ModelBrowser/PresetModelListWidget.h</header>
</customwidget>
</customwidgets>
<resources/> <resources/>
<connections/> <connections/>
</ui> </ui>

View File

@ -191,10 +191,14 @@ void OsgWidget::paintEvent(QPaintEvent*) {
} }
void OsgWidget::dragEnterEvent(QDragEnterEvent* event) { void OsgWidget::dragEnterEvent(QDragEnterEvent* event) {
// 检查是否是预制模型拖拽 LOG_INFO("OsgWidget::dragEnterEvent - Available formats:{}", event->mimeData()->formats().join(",").toStdString());
// Check if it's a preset model drag
if (event->mimeData()->hasFormat("application/x-preset-model")) { if (event->mimeData()->hasFormat("application/x-preset-model")) {
LOG_INFO("OsgWidget::dragEnterEvent - Accept preset model drag");
event->acceptProposedAction(); event->acceptProposedAction();
} else { } else {
LOG_INFO("OsgWidget::dragEnterEvent - Ignore drag, not preset model format");
event->ignore(); event->ignore();
} }
} }
@ -209,28 +213,33 @@ void OsgWidget::dragMoveEvent(QDragMoveEvent* event) {
} }
void OsgWidget::dropEvent(QDropEvent* event) { void OsgWidget::dropEvent(QDropEvent* event) {
// 检查是否是预制模型拖拽 LOG_INFO("OsgWidget::dropEvent - Start processing drop event");
if (event->mimeData()->hasFormat("application/x-preset-model")) { if (event->mimeData()->hasFormat("application/x-preset-model")) {
QByteArray data = event->mimeData()->data("application/x-preset-model"); QByteArray data = event->mimeData()->data("application/x-preset-model");
QString modelInfo = QString::fromUtf8(data); QString modelInfo = QString::fromUtf8(data);
// 解析模型信息 (格式: "modelType|modelName") LOG_INFO("OsgWidget::dropEvent - Received model info:{}", modelInfo.toStdString());
QStringList parts = modelInfo.split("|"); QStringList parts = modelInfo.split("|");
if (parts.size() == 2) { if (parts.size() == 2) {
QString modelType = parts[0]; QString modelType = parts[0];
QString modelName = parts[1]; QString modelName = parts[1];
// 获取拖拽位置
QPointF position = event->posF(); QPointF position = event->posF();
// 处理预制模型拖拽 LOG_INFO("OsgWidget::dropEvent - Parse success, type:{} name:{} position:({},{})", modelType.toStdString(), modelName.toStdString(), position.x(), position.y());
OnPresetModelDropped(modelType, modelName, position); OnPresetModelDropped(modelType, modelName, position);
event->acceptProposedAction(); event->acceptProposedAction();
LOG_INFO("OsgWidget::dropEvent - Drag processing completed, accepted");
} else { } else {
LOG_ERROR("OsgWidget::dropEvent - Data format error, expected: type|name, actual:{}", modelInfo.toStdString());
event->ignore(); event->ignore();
} }
} else { } else {
LOG_INFO("OsgWidget::dropEvent - Not preset model format, ignore drag");
event->ignore(); event->ignore();
} }
} }
@ -241,7 +250,7 @@ void OsgWidget::OnPresetModelDropped(const QString& modelType, const QString& mo
// 2. 创建对应的Entity和MeshComponent // 2. 创建对应的Entity和MeshComponent
// 3. 添加到场景中 // 3. 添加到场景中
qDebug() << "预制模型拖拽:" << modelType << modelName << "位置:" << position; LOG_INFO("OsgWidget::OnPresetModelDropped - Dropped model type:{} name:{} position:({}, {})", modelType.toStdString(), modelName.toStdString(), position.x(), position.y());
// 这里暂时只是输出调试信息,后续需要集成实体系统 // 这里暂时只是输出调试信息,后续需要集成实体系统
} }

View File

@ -3,21 +3,11 @@
<model name="Electronic Jammer" displayName="Electronic Jammer" <model name="Electronic Jammer" displayName="Electronic Jammer"
description="Comprehensive electronic warfare jamming equipment" description="Comprehensive electronic warfare jamming equipment"
meshFile="models/electronic_jammer.ive" meshFile="models/electronic_jammer.ive"
enabled="true" /> icon="icons/electronic_jammer_icon.png"
<model name="Communication Jammer" displayName="Communication Jammer"
description="Specialized communication signal jamming equipment"
meshFile="models/comm_jammer.ive"
enabled="true" />
<model name="Radar Jammer" displayName="Radar Jammer"
description="Radar signal jamming and deception equipment"
meshFile="models/radar_jammer.ive"
enabled="true" />
<model name="GPS Jammer" displayName="GPS Jammer"
description="Satellite navigation signal jamming equipment"
meshFile="models/gps_jammer.ive"
enabled="true" /> enabled="true" />
<model name="Signal Jammer" displayName="Signal Jammer" <model name="Signal Jammer" displayName="Signal Jammer"
description="Multi-frequency signal jamming equipment" description="Multi-frequency signal jamming equipment"
meshFile="models/signal_jammer.ive" meshFile="models/signal_jammer.ive"
icon="icons/signal_jammer_icon.png"
enabled="true" /> enabled="true" />
</modelCategory> </modelCategory>

View File

@ -1,23 +1,8 @@
<?xml version="1.0" encoding="UTF-8"?> <?xml version="1.0" encoding="UTF-8"?>
<modelCategory name="Missile" displayName="Missiles" icon="missile_icon.png"> <modelCategory name="Missile" displayName="Missiles" icon="missile_icon.png">
<model name="Cruise Missile" displayName="Cruise Missile"
description="Long-range precision strike cruise missile"
meshFile="models/cruise_missile.ive"
enabled="true" />
<model name="Ballistic Missile" displayName="Ballistic Missile" <model name="Ballistic Missile" displayName="Ballistic Missile"
description="Long-range ballistic missile system" description="Long-range ballistic missile system"
meshFile="models/ballistic_missile.ive" meshFile="models/ballistic_missile.ive"
enabled="true" /> icon="icons/ballistic_missile_icon.png"
<model name="Anti-Ship Missile" displayName="Anti-Ship Missile"
description="Specialized anti-ship attack missile"
meshFile="models/antiship_missile.ive"
enabled="true" />
<model name="Surface-to-Air Missile" displayName="Surface-to-Air Missile"
description="Surface-to-air defense missile system"
meshFile="models/sam_missile.ive"
enabled="true" />
<model name="Anti-Tank Missile" displayName="Anti-Tank Missile"
description="Anti-armor precision guided missile"
meshFile="models/antitank_missile.ive"
enabled="true" /> enabled="true" />
</modelCategory> </modelCategory>

View File

@ -1,23 +1,8 @@
<?xml version="1.0" encoding="UTF-8"?> <?xml version="1.0" encoding="UTF-8"?>
<modelCategory name="Satellite" displayName="Satellites" icon="satellite_icon.png"> <modelCategory name="Satellite" displayName="Satellites" icon="satellite_icon.png">
<model name="Communication Satellite" displayName="Communication Satellite" <model name="Communication Satellite" displayName="Communication Satellite"
description="Satellite platform providing global communication services" description="High-capacity communication relay satellite"
meshFile="models/comm_satellite.ive" meshFile="models/comm_satellite.ive"
enabled="true" /> icon="icons/comm_satellite_icon.png"
<model name="Weather Satellite" displayName="Weather Satellite"
description="Observation satellite monitoring weather and climate changes"
meshFile="models/weather_satellite.ive"
enabled="true" />
<model name="GPS Satellite" displayName="GPS Satellite"
description="Global Positioning System satellite"
meshFile="models/gps_satellite.ive"
enabled="true" />
<model name="Spy Satellite" displayName="Spy Satellite"
description="Military reconnaissance and surveillance satellite"
meshFile="models/spy_satellite.ive"
enabled="true" />
<model name="Scientific Satellite" displayName="Scientific Satellite"
description="Dedicated satellite for scientific research"
meshFile="models/science_satellite.ive"
enabled="true" /> enabled="true" />
</modelCategory> </modelCategory>

View File

@ -3,21 +3,11 @@
<model name="Destroyer" displayName="Destroyer" <model name="Destroyer" displayName="Destroyer"
description="High-speed combat vessel with powerful anti-submarine and air defense capabilities" description="High-speed combat vessel with powerful anti-submarine and air defense capabilities"
meshFile="models/destroyer.ive" meshFile="models/destroyer.ive"
icon="icons/destroyer_icon.png"
enabled="true" /> enabled="true" />
<model name="Aircraft Carrier" displayName="Aircraft Carrier" <model name="Aircraft Carrier" displayName="Aircraft Carrier"
description="Large aircraft carrier, mobile airfield and command center at sea" description="Large aircraft carrier, mobile airfield and command center at sea"
meshFile="models/carrier.ive" meshFile="models/carrier.ive"
enabled="true" /> icon="icons/carrier_icon.png"
<model name="Frigate" displayName="Frigate"
description="Medium-sized escort vessel, multi-purpose combat platform"
meshFile="models/frigate.ive"
enabled="true" />
<model name="Submarine" displayName="Submarine"
description="Underwater combat platform with strong stealth capabilities"
meshFile="models/submarine.ive"
enabled="true" />
<model name="Cruiser" displayName="Cruiser"
description="Heavy combat vessel with powerful firepower"
meshFile="models/cruiser.ive"
enabled="true" /> enabled="true" />
</modelCategory> </modelCategory>