GUI evolutions

This commit is contained in:
Sylvain Schneider
2026-03-17 11:53:20 +01:00
parent 03d2e94f8b
commit d48abfba79
5 changed files with 152 additions and 122 deletions

View File

@@ -23,8 +23,10 @@ MainFrame::MainFrame(dBus::Bus &bus)
// Initialization // Initialization
SetIcon(wxICON(AAAA_ICON)); SetIcon(wxICON(AAAA_ICON));
const wxSize minSize(1200, 900); const wxSize minSize(1024, 768);
const wxSize maxSize(1200, -1);
SetMinSize(minSize); SetMinSize(minSize);
SetMaxSize(maxSize);
SetSize(minSize); SetSize(minSize);
// Creating controls // Creating controls

View File

@@ -1,15 +1,16 @@
#include "requirementDetailPanel.h" #include "requirementDetailPanel.h"
#include <wx/statline.h>
using namespace std; using namespace std;
using namespace gui; using namespace gui;
//-------------------------------------------------------------- //--------------------------------------------------------------
/* Constructor */ /* Constructor */
RequirementDetailPanel::RequirementDetailPanel(wxWindow *parentWindow, dBus::Bus &bus) RequirementDetailPanel::RequirementDetailPanel(wxWindow *parentWindow, dBus::Bus &bus)
: wxScrolledWindow(parentWindow, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxSIMPLE_BORDER | wxVSCROLL) : wxPanel(parentWindow /*, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxSIMPLE_BORDER*/)
, m_bus(bus) , m_bus(bus)
{ {
// Initialization // Initialization
SetScrollRate(0, 10); // Set the scroll rate for the panel (vertical only)
// Creating controls // Creating controls
createControls(); createControls();
@@ -20,24 +21,35 @@ RequirementDetailPanel::RequirementDetailPanel(wxWindow *parentWindow, dBus::Bus
/* Creating controls */ /* Creating controls */
void RequirementDetailPanel::createControls() void RequirementDetailPanel::createControls()
{ {
const auto mainPanel = new wxPanel(this, wxID_ANY, wxDefaultPosition, wxSize(500, -1), wxSIMPLE_BORDER); const auto mainPanel = new wxScrolledWindow(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxVSCROLL);
mainPanel->SetScrollRate(0, 10); // Set the scroll rate for the panel (vertical only)
const auto closeButton = new wxButton(this, wxID_ANY, _("Close"));
const auto saveButton = new wxButton(this, wxID_ANY, _("Save"));
const auto discardButton = new wxButton(this, wxID_ANY, _("Discard"));
// Controls positioning // Controls positioning
const auto ctrlSizer = new wxBoxSizer(wxVERTICAL); const auto ctrlSizer = new wxBoxSizer(wxVERTICAL);
ctrlSizer->Add(createControls_metadata(mainPanel), wxSizerFlags(0).Expand()); ctrlSizer->Add(createControls_metadata(mainPanel), wxSizerFlags(0).Expand().Border(wxALL & ~wxTOP, 5));
ctrlSizer->Add(createControls_details(mainPanel), wxSizerFlags(0).Expand()); ctrlSizer->Add(createControls_details(mainPanel), wxSizerFlags(0).Expand().Border(wxALL & ~wxTOP, 5));
ctrlSizer->Add(createControls_classification(mainPanel), wxSizerFlags(0).Expand()); ctrlSizer->Add(createControls_classification(mainPanel), wxSizerFlags(0).Expand().Border(wxALL & ~wxTOP & ~wxBOTTOM, 5));
mainPanel->SetSizer(ctrlSizer); mainPanel->SetSizer(ctrlSizer);
const auto buttonSizer = new wxBoxSizer(wxVERTICAL);
buttonSizer->Add(closeButton, wxSizerFlags(0).Expand());
buttonSizer->AddStretchSpacer(1);
buttonSizer->Add(discardButton, wxSizerFlags(0).Expand().Border(wxBOTTOM, 5));
buttonSizer->Add(saveButton, wxSizerFlags(0).Expand());
const auto mainSizer = new wxBoxSizer(wxHORIZONTAL); const auto mainSizer = new wxBoxSizer(wxHORIZONTAL);
mainSizer->Add(mainPanel, wxSizerFlags(0).Expand()); mainSizer->Add(mainPanel, wxSizerFlags(1).Expand().Border(wxLEFT | wxRIGHT, 5));
mainSizer->AddStretchSpacer(1); mainSizer->Add(buttonSizer, wxSizerFlags(0).Expand().Border(wxALL, 5));
SetSizer(mainSizer); SetSizer(mainSizer);
// Set the virtual size to match the size of the child panel, // Set the virtual size to match the size of the child panel,
// enabling scrolling if necessary // enabling scrolling if necessary
FitInside(); mainPanel->FitInside();
// Harmonize the sizes of the title controls to ensure they have the same width // Harmonize the sizes of the title controls to ensure they have the same width
updateTitleSizes(); updateTitleSizes();
@@ -47,46 +59,30 @@ void RequirementDetailPanel::createControls()
/* Creating controls for requirement metadata (ID, UUID, name, description, etc.) */ /* Creating controls for requirement metadata (ID, UUID, name, description, etc.) */
wxSizer *RequirementDetailPanel::createControls_metadata(wxWindow *owner) wxSizer *RequirementDetailPanel::createControls_metadata(wxWindow *owner)
{ {
struct Item const auto uuidTitleCtrl = createTitle(owner, _T("UUID:"));
{ const auto idTitleCtrl = createTitle(owner, _T("ID:"));
wxWindow *titleCtrl; const auto authorTitleCtrl = createTitle(owner, _("Author:"));
wxWindow *inputCtrl; const auto createdAtTitleCtrl = createTitle(owner, _("Created at:"));
}; const auto updatedAtTitleCtrl = createTitle(owner, _("Updated at:"));
std::vector<Item> items;
{ const auto uuidValueCtrl = new wxTextCtrl(owner, wxID_ANY);
const Item item{ .titleCtrl = createTitle(owner, _T("UUID:")), const auto idValueCtrl = new wxTextCtrl(owner, wxID_ANY);
.inputCtrl = new wxTextCtrl(owner, wxID_ANY) }; const auto authorValueCtrl = new wxTextCtrl(owner, wxID_ANY);
items.push_back(item); const auto createdAtValueCtrl = new wxTextCtrl(owner, wxID_ANY);
} const auto updatedAtValueCtrl = new wxTextCtrl(owner, wxID_ANY);
{
const Item item{ .titleCtrl = createTitle(owner, _T("ID:")),
.inputCtrl = new wxTextCtrl(owner, wxID_ANY) };
items.push_back(item);
}
{
const Item item{ .titleCtrl = createTitle(owner, _("Author:")),
.inputCtrl = new wxTextCtrl(owner, wxID_ANY) };
items.push_back(item);
}
{
const Item item{ .titleCtrl = createTitle(owner, _("Created at:")),
.inputCtrl = new wxTextCtrl(owner, wxID_ANY) };
items.push_back(item);
}
{
const Item item{ .titleCtrl = createTitle(owner, _("Updated at:")),
.inputCtrl = new wxTextCtrl(owner, wxID_ANY) };
items.push_back(item);
}
// Controls positioning // Controls positioning
const auto ctrlSizer = new wxFlexGridSizer(2, 5, 5); const auto ctrlSizer = new wxFlexGridSizer(2, 5, 5);
for (const auto &item : items) ctrlSizer->Add(uuidTitleCtrl, wxSizerFlags(0).Expand().CenterVertical());
{ ctrlSizer->Add(uuidValueCtrl, wxSizerFlags(1).Expand());
ctrlSizer->Add(item.titleCtrl, wxSizerFlags(0).Expand().CenterVertical()); ctrlSizer->Add(idTitleCtrl, wxSizerFlags(0).Expand().CenterVertical());
ctrlSizer->Add(item.inputCtrl, wxSizerFlags(1).Expand()); ctrlSizer->Add(idValueCtrl, wxSizerFlags(1).Expand());
} ctrlSizer->Add(authorTitleCtrl, wxSizerFlags(0).Expand().CenterVertical());
ctrlSizer->Add(authorValueCtrl, wxSizerFlags(1).Expand());
ctrlSizer->Add(createdAtTitleCtrl, wxSizerFlags(0).Expand().CenterVertical());
ctrlSizer->Add(createdAtValueCtrl, wxSizerFlags(1).Expand());
ctrlSizer->Add(updatedAtTitleCtrl, wxSizerFlags(0).Expand().CenterVertical());
ctrlSizer->Add(updatedAtValueCtrl, wxSizerFlags(1).Expand());
ctrlSizer->AddGrowableCol(1, 1); ctrlSizer->AddGrowableCol(1, 1);
// Controls positioning // Controls positioning
@@ -99,55 +95,28 @@ wxSizer *RequirementDetailPanel::createControls_metadata(wxWindow *owner)
/* Creating controls for requirement details (e.g., custom attributes, child requirements, etc.) */ /* Creating controls for requirement details (e.g., custom attributes, child requirements, etc.) */
wxSizer *RequirementDetailPanel::createControls_details(wxWindow *owner) wxSizer *RequirementDetailPanel::createControls_details(wxWindow *owner)
{ {
struct Item const auto titleTitleCtrl = createTitle(owner, _("Title:"));
{ const auto descriptionTitleCtrl = createTitle(owner, _("Description:"));
wxWindow *titleCtrl; const auto acceptanceCriteriaTitleCtrl = createTitle(owner, _("Acceptance criteria:"));
wxWindow *inputCtrl;
};
std::vector<Item> items;
{ const auto titleValueCtrl = new wxTextCtrl(owner, wxID_ANY);
const Item item{ .titleCtrl = createTitle(owner, _("Title:")), const auto descriptionValueCtrl = new wxTextCtrl(owner, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_PROCESS_TAB | wxTE_MULTILINE);
.inputCtrl = new wxTextCtrl(owner, wxID_ANY) }; const auto acceptanceCriteriaValueCtrl = new wxTextCtrl(owner, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_PROCESS_TAB | wxTE_MULTILINE);
items.push_back(item); const auto smartCheckBoxCtrl = new wxCheckBox(owner, wxID_ANY, _("S.M.A.R.T."));
}
{
const Item item{ .titleCtrl = createTitle(owner, _("Description:")),
.inputCtrl = new wxTextCtrl(owner,
wxID_ANY,
wxEmptyString,
wxDefaultPosition,
wxDefaultSize,
wxTE_PROCESS_TAB | wxTE_MULTILINE | wxVSCROLL | wxTE_AUTO_URL) };
items.push_back(item);
item.inputCtrl->SetMinSize(wxSize(-1, 200)); descriptionValueCtrl->SetMinSize(wxSize(-1, 200));
} acceptanceCriteriaValueCtrl->SetMinSize(wxSize(-1, 100));
{
const Item item{ .titleCtrl = createTitle(owner, _("Acceptance criteria:")),
.inputCtrl = new wxTextCtrl(owner,
wxID_ANY,
wxEmptyString,
wxDefaultPosition,
wxDefaultSize,
wxTE_PROCESS_TAB | wxTE_MULTILINE | wxVSCROLL) };
items.push_back(item);
item.inputCtrl->SetMinSize(wxSize(-1, 100));
}
{
const Item item{ .titleCtrl = new wxPanel(owner),
.inputCtrl = new wxCheckBox(owner, wxID_ANY, _("S.M.A.R.T.")) };
items.push_back(item);
}
// Controls positioning // Controls positioning
const auto ctrlSizer = new wxFlexGridSizer(2, 5, 5); const auto ctrlSizer = new wxFlexGridSizer(2, 5, 5);
for (const auto &item : items) ctrlSizer->Add(titleTitleCtrl, wxSizerFlags(0).Expand().CenterVertical());
{ ctrlSizer->Add(titleValueCtrl, wxSizerFlags(1).Expand());
ctrlSizer->Add(item.titleCtrl, wxSizerFlags(0).Expand().CenterVertical()); ctrlSizer->Add(descriptionTitleCtrl, wxSizerFlags(0).Expand());
ctrlSizer->Add(item.inputCtrl, wxSizerFlags(1).Expand()); ctrlSizer->Add(descriptionValueCtrl, wxSizerFlags(1).Expand());
} ctrlSizer->Add(acceptanceCriteriaTitleCtrl, wxSizerFlags(0).Expand());
ctrlSizer->Add(acceptanceCriteriaValueCtrl, wxSizerFlags(1).Expand());
ctrlSizer->AddStretchSpacer();
ctrlSizer->Add(smartCheckBoxCtrl, wxSizerFlags(1).Expand());
ctrlSizer->AddGrowableCol(1, 1); ctrlSizer->AddGrowableCol(1, 1);
const auto localSizer = new wxStaticBoxSizer(wxVERTICAL, owner, "Details"); const auto localSizer = new wxStaticBoxSizer(wxVERTICAL, owner, "Details");
@@ -159,41 +128,26 @@ wxSizer *RequirementDetailPanel::createControls_details(wxWindow *owner)
/* Creating controls for requirement classification (e.g., priority, severity, etc.) */ /* Creating controls for requirement classification (e.g., priority, severity, etc.) */
wxSizer *RequirementDetailPanel::createControls_classification(wxWindow *owner) wxSizer *RequirementDetailPanel::createControls_classification(wxWindow *owner)
{ {
struct Item const auto typeTitleCtrl = createTitle(owner, _("Type:"));
{ const auto categoryTitleCtrl = createTitle(owner, _("Category:"));
wxWindow *titleCtrl; const auto priorityTitleCtrl = createTitle(owner, _("Priority:"));
wxWindow *inputCtrl; const auto statusTitleCtrl = createTitle(owner, _("Status:"));
};
std::vector<Item> items;
{ const auto typeValueCtrl = new wxTextCtrl(owner, wxID_ANY);
const Item item{ .titleCtrl = createTitle(owner, "Type:"), const auto categoryValueCtrl = new wxTextCtrl(owner, wxID_ANY);
.inputCtrl = new wxTextCtrl(owner, wxID_ANY) }; const auto priorityValueCtrl = new wxTextCtrl(owner, wxID_ANY);
items.push_back(item); const auto statusValueCtrl = new wxTextCtrl(owner, wxID_ANY);
}
{
const Item item{ .titleCtrl = createTitle(owner, "Category:"),
.inputCtrl = new wxTextCtrl(owner, wxID_ANY) };
items.push_back(item);
}
{
const Item item{ .titleCtrl = createTitle(owner, "Priority:"),
.inputCtrl = new wxTextCtrl(owner, wxID_ANY) };
items.push_back(item);
}
{
const Item item{ .titleCtrl = createTitle(owner, "Status:"),
.inputCtrl = new wxTextCtrl(owner, wxID_ANY) };
items.push_back(item);
}
// Controls positioning // Controls positioning
const auto ctrlSizer = new wxFlexGridSizer(2, 5, 5); const auto ctrlSizer = new wxFlexGridSizer(2, 5, 5);
for (const auto &item : items) ctrlSizer->Add(typeTitleCtrl, wxSizerFlags(0).Expand().CenterVertical());
{ ctrlSizer->Add(typeValueCtrl, wxSizerFlags(1).Expand());
ctrlSizer->Add(item.titleCtrl, wxSizerFlags(0).Expand().CenterVertical()); ctrlSizer->Add(categoryTitleCtrl, wxSizerFlags(0).Expand().CenterVertical());
ctrlSizer->Add(item.inputCtrl, wxSizerFlags(1).Expand()); ctrlSizer->Add(categoryValueCtrl, wxSizerFlags(1).Expand());
} ctrlSizer->Add(priorityTitleCtrl, wxSizerFlags(0).Expand().CenterVertical());
ctrlSizer->Add(priorityValueCtrl, wxSizerFlags(1).Expand());
ctrlSizer->Add(statusTitleCtrl, wxSizerFlags(0).Expand().CenterVertical());
ctrlSizer->Add(statusValueCtrl, wxSizerFlags(1).Expand());
ctrlSizer->AddGrowableCol(1, 1); ctrlSizer->AddGrowableCol(1, 1);
const auto localSizer = new wxStaticBoxSizer(wxVERTICAL, owner, "Classification"); const auto localSizer = new wxStaticBoxSizer(wxVERTICAL, owner, "Classification");
@@ -211,6 +165,7 @@ wxStaticText *RequirementDetailPanel::createTitle(wxWindow *owner, const wxStrin
wxDefaultPosition, wxDefaultPosition,
wxDefaultSize, wxDefaultSize,
wxST_NO_AUTORESIZE | wxALIGN_RIGHT); wxST_NO_AUTORESIZE | wxALIGN_RIGHT);
ctrl->SetFont(GetFont().Italic());
m_titleControls.push_back(ctrl); m_titleControls.push_back(ctrl);

View File

@@ -6,7 +6,7 @@
namespace gui namespace gui
{ {
//-------------------------------------------------------------- //--------------------------------------------------------------
class RequirementDetailPanel : public wxScrolledWindow class RequirementDetailPanel : public wxPanel
{ {
public: public:
RequirementDetailPanel() = delete; // Default constructor RequirementDetailPanel() = delete; // Default constructor

View File

@@ -0,0 +1,72 @@
#pragma once
#include <filesystem>
#include <wx/dataview.h>
#include <wx/wx.h>
namespace gui
{
//--------------------------------------------------------------
struct RequirementData
{
std::string uuid;
std::string id;
std::string title;
RequirementData *parent;
std::vector<std::unique_ptr<RequirementData>> children;
RequirementData(std::string uuid_, std::string id_, std::string title_)
: uuid(std::move(uuid_))
, id(std::move(id_))
, title(std::move(title_))
{
}
// Adds a child requirement
RequirementData *AddChild(std::unique_ptr<RequirementData> child)
{
child->parent = this;
children.push_back(std::move(child));
return children.back().get();
}
};
//--------------------------------------------------------------
/* --- */
//--------------------------------------------------------------
class RequirementTreeModel : public wxDataViewModel
{
public:
RequirementTreeModel(); // Default constructor
virtual ~RequirementTreeModel() = default; // Default destructor
RequirementTreeModel(const RequirementTreeModel &obj) = delete; // Copy constructor
RequirementTreeModel(RequirementTreeModel &&obj) noexcept = delete; // Move constructor
RequirementTreeModel &operator=(const RequirementTreeModel &obj) = delete; // Copy assignment operator
RequirementTreeModel &operator=(RequirementTreeModel &&obj) noexcept = delete; // Move assignment operator
// wxDataViewModel overrides methods
unsigned GetColumnCount() const override;
wxString GetColumnType(unsigned) const override;
void GetValue(wxVariant &variant, const wxDataViewItem &item, unsigned col) const override;
bool SetValue(const wxVariant &variant, const wxDataViewItem &item, unsigned col) override;
wxDataViewItem GetParent(const wxDataViewItem &item) const override;
bool IsContainer(const wxDataViewItem &item) const override;
bool HasContainerColumns(const wxDataViewItem &) const override;
unsigned GetChildren(const wxDataViewItem &item, wxDataViewItemArray &children) const override;
// Tree data model specific methods
RequirementData *AddRootRequirement(std::unique_ptr<RequirementData> requirementData); // Add a root requirement to the tree
RequirementData *AddChildRequirement(RequirementData *parent, std::unique_ptr<RequirementData> requirementData); // Add a child requirement to a parent requirement
void clear();
RequirementData *GetRequirementData(const wxDataViewItem &item) const; // Get the requirement data associated with a given item
protected:
std::vector<std::unique_ptr<RequirementData>> m_rootRequirement;
};
//--------------------------------------------------------------
} // namespace gui

View File

@@ -29,6 +29,7 @@ void RequirementsPanel::createControls()
const auto mainSizer = new wxBoxSizer(wxHORIZONTAL); const auto mainSizer = new wxBoxSizer(wxHORIZONTAL);
mainSizer->Add(m_requirementListPanel, wxSizerFlags(0).Expand().Border(wxALL, 5)); mainSizer->Add(m_requirementListPanel, wxSizerFlags(0).Expand().Border(wxALL, 5));
mainSizer->Add(m_requirementDetailPanel, wxSizerFlags(1).Expand().Border(wxALL, 5)); mainSizer->Add(m_requirementDetailPanel, wxSizerFlags(1).Expand().Border(wxALL, 5));
//mainSizer->AddStretchSpacer(1);
SetSizer(mainSizer); SetSizer(mainSizer);
} }