From fc4beb9f3a24610a395f5b4a54a8f87a50c66fda Mon Sep 17 00:00:00 2001 From: archibalduk Date: Sat, 15 Jun 2024 13:20:04 +0100 Subject: [PATCH] Add example: Simple Table View --- README.md | 2 +- examples/01-01_Simple_Table_View/.gitignore | 78 ++++++++ .../01-01_Simple_Table_View.pro | 34 ++++ examples/01-01_Simple_Table_View/README.md | 22 +++ examples/01-01_Simple_Table_View/main.cpp | 22 +++ .../01-01_Simple_Table_View/mainwindow.cpp | 175 ++++++++++++++++++ examples/01-01_Simple_Table_View/mainwindow.h | 45 +++++ 7 files changed, 377 insertions(+), 1 deletion(-) create mode 100644 examples/01-01_Simple_Table_View/.gitignore create mode 100644 examples/01-01_Simple_Table_View/01-01_Simple_Table_View.pro create mode 100644 examples/01-01_Simple_Table_View/README.md create mode 100644 examples/01-01_Simple_Table_View/main.cpp create mode 100644 examples/01-01_Simple_Table_View/mainwindow.cpp create mode 100644 examples/01-01_Simple_Table_View/mainwindow.h diff --git a/README.md b/README.md index 9bbee3b..afaddc8 100644 --- a/README.md +++ b/README.md @@ -41,7 +41,7 @@ Data can be accessed using [QModelIndex](https://doc.qt.io/qt-6/qmodelindex.html ## Author [@archibalduk](https://www.github.com/archibalduk) -# Credits +## Credits EHM DAL uses [QXlsx by j2doll](https://github.com/QtExcel/QXlsx). QXlsx is licensed under the MIT license. ## License diff --git a/examples/01-01_Simple_Table_View/.gitignore b/examples/01-01_Simple_Table_View/.gitignore new file mode 100644 index 0000000..3fe56d4 --- /dev/null +++ b/examples/01-01_Simple_Table_View/.gitignore @@ -0,0 +1,78 @@ +# This file is used to ignore files which are generated +# ---------------------------------------------------------------------------- + +*~ +*.autosave +*.a +*.core +*.moc +*.o +*.obj +*.orig +*.rej +*.so +*.so.* +*_pch.h.cpp +*_resource.rc +*.qm +.#* +*.*# +core +!core/ +tags +.DS_Store +.directory +*.debug +Makefile* +*.prl +*.app +moc_*.cpp +ui_*.h +qrc_*.cpp +Thumbs.db +*.res +*.rc +/.qmake.cache +/.qmake.stash + +# qtcreator generated files +*.pro.user* +CMakeLists.txt.user* + +# xemacs temporary files +*.flc + +# Vim temporary files +.*.swp + +# Visual Studio generated files +*.ib_pdb_index +*.idb +*.ilk +*.pdb +*.sln +*.suo +*.vcproj +*vcproj.*.*.user +*.ncb +*.sdf +*.opensdf +*.vcxproj +*vcxproj.* + +# MinGW generated files +*.Debug +*.Release + +# Python byte code +*.pyc + +# Binaries +# -------- +*.dll +*.exe + +# DAL project files +/build/ +/lib/ehm_dal/ +/lib/qxlsx/ \ No newline at end of file diff --git a/examples/01-01_Simple_Table_View/01-01_Simple_Table_View.pro b/examples/01-01_Simple_Table_View/01-01_Simple_Table_View.pro new file mode 100644 index 0000000..804d4a2 --- /dev/null +++ b/examples/01-01_Simple_Table_View/01-01_Simple_Table_View.pro @@ -0,0 +1,34 @@ +QT += core gui + +greaterThan(QT_MAJOR_VERSION, 4): QT += widgets + +CONFIG += c++latest + +# The following define makes your compiler emit warnings if you use +# any Qt feature that has been marked deprecated (the exact warnings +# depend on your compiler). Please consult the documentation of the +# deprecated API in order to know how to port your code away from it. +DEFINES += QT_DEPRECATED_WARNINGS + +# You can make your code fail to compile if it uses deprecated APIs. +# In order to do so, uncomment the following line. +DEFINES += QT_DISABLE_DEPRECATED_BEFORE=0x060000 # disables all the APIs deprecated before Qt 6.0.0 + +# Enable as many compiler warnings as possible +CONFIG += warn_on + +# EHM DAL +include($$PWD/lib/ehm_dal/ehm_dal.pri) +include($$PWD/lib/qxlsx/qxlsx.pri) + +SOURCES += \ + main.cpp \ + mainwindow.cpp + +HEADERS += \ + mainwindow.h + +# Default rules for deployment. +qnx: target.path = /tmp/$${TARGET}/bin +else: unix:!android: target.path = /opt/$${TARGET}/bin +!isEmpty(target.path): INSTALLS += target diff --git a/examples/01-01_Simple_Table_View/README.md b/examples/01-01_Simple_Table_View/README.md new file mode 100644 index 0000000..35c2335 --- /dev/null +++ b/examples/01-01_Simple_Table_View/README.md @@ -0,0 +1,22 @@ +# EHM DAL: Example 01.01 - Simple Table View +The purpose of this example is to show how easily data from either a database or a saved game's database can be shown using QTableView and exported to csv/xslx spreadsheets. + +## How to compile the project +1. Download the source code from this example project folder and save to your PC. +1. Create a new sub-folder named `lib` within your downloaded folder. +1. Download the latest version of EHM DAL from the [Releases page](https://github.com/archibalduk/EHM_DAL/releases) and extract the contents of the downloaded .zip into your lib folder. +1. Open the `.pro` file in Qt Creator. +1. Press `Ctrl + R` in Qt Creator to build and run the application. + +**Visit the [EHM DAL documentation](https://archibalduk.github.io/EHM_DAL/annotated.html) for details of the various classes and functions.** + +**Visit the [EHM DAL wiki](https://github.com/archibalduk/EHM_DAL/wiki) for guides and tutorials.** + +## Author +[@archibalduk](https://www.github.com/archibalduk) + +## Credits +EHM DAL uses [QXlsx by j2doll](https://github.com/QtExcel/QXlsx). QXlsx is licensed under the MIT license. + +## License +[GNU General Public License v3.0](https://choosealicense.com/licenses/gpl-3.0/) diff --git a/examples/01-01_Simple_Table_View/main.cpp b/examples/01-01_Simple_Table_View/main.cpp new file mode 100644 index 0000000..b2ed2d2 --- /dev/null +++ b/examples/01-01_Simple_Table_View/main.cpp @@ -0,0 +1,22 @@ +#include "mainwindow.h" + +#include +#include + +int main(int argc, char *argv[]) +{ + // Application + QApplication a(argc, argv); + + // Use Fusion style if Windows dark mode is enabled (Fusion is dark mode compatible) + QSettings settings( + "HKEY_CURRENT_USER\\Software\\Microsoft\\Windows\\CurrentVersion\\Themes\\Personalize", + QSettings::NativeFormat); + if (settings.value("AppsUseLightTheme") == 0) + a.setStyle("fusion"); + + // Main window + MainWindow w; + w.show(); + return a.exec(); +} diff --git a/examples/01-01_Simple_Table_View/mainwindow.cpp b/examples/01-01_Simple_Table_View/mainwindow.cpp new file mode 100644 index 0000000..d6896f9 --- /dev/null +++ b/examples/01-01_Simple_Table_View/mainwindow.cpp @@ -0,0 +1,175 @@ +#include "mainwindow.h" + +#include "lib/ehm_dal/include/database/column_data/club_competition_columns.h" +#include "lib/ehm_dal/include/database/database.h" +#include "lib/ehm_dal/include/database/file_io/database_file_manager.h" +#include "lib/ehm_dal/include/saved_game/file_io/saved_game_file_manager.h" +#include "lib/ehm_dal/include/spreadsheet/spreadsheet.h" +#include "lib/ehm_dal/include/tables/abstract_table.h" +#include "lib/ehm_dal/include/tables/table_index.h" + +#include +#include + +/* MAIN WINDOW CONSTRUCTOR + * This function constructs the main window and adds the following to the window: + * 1. A menu bar with a drop-down menu allowing users to open a database or saved game; and + * 2. A QTableView widget which will show the selected database table within the window. + */ +MainWindow::MainWindow(QWidget *parent) + : QMainWindow(parent) +{ + /// Window attributes /// + setMinimumSize(1200, 600); + setWindowTitle("Simple Table Viewer"); + + /// Menu bar /// + menu_bar_ = new QMenuBar(this); + setMenuBar(menu_bar_); + + /// Menu bar -> File menu /// + auto file_menu{menu_bar_->addMenu("&File")}; + + // Create a new QAction which will be shown in the File menu. + // Clicking on this QAction will trigger the openDatabase() function. + open_database_action_ = new QAction("Open &database...", this); + QObject::connect(open_database_action_, &QAction::triggered, this, &MainWindow::openDatabase); + file_menu->addAction(open_database_action_); + + // Create a new QAction which will be shown in the File menu. + // Clicking on this QAction will trigger the openSavedGame() function. + open_saved_game_action_ = new QAction("Open saved &game...", this); + QObject::connect(open_saved_game_action_, &QAction::triggered, this, &MainWindow::openSavedGame); + file_menu->addAction(open_saved_game_action_); + + /// Menu bar -> Tools menu /// + auto tools_menu{menu_bar_->addMenu("&Tools")}; + + // Create a new QAction which will be shown in the Tools menu. + // Clicking on this QAction will trigger the exportSpreadsheet() function. + spreadsheet_export_all_columns_action_ = new QAction("&Export to spreadsheet...", this); + QObject::connect(spreadsheet_export_all_columns_action_, + &QAction::triggered, + this, + &MainWindow::exportSpreadsheet); + tools_menu->addAction(spreadsheet_export_all_columns_action_); + + // Create a new QAction which will be shown in the Tools menu. + // Clicking on this QAction will trigger the exportSpreadsheetSelectedColumns() function. + spreadsheet_export_selected_columns_action_ + = new QAction("Export to spreadsheet (selected &columns)...", this); + QObject::connect(spreadsheet_export_selected_columns_action_, + &QAction::triggered, + this, + &MainWindow::exportSpreadsheetSelectedColumns); + tools_menu->addAction(spreadsheet_export_selected_columns_action_); + + /// Table view /// + table_view_ = new QTableView(this); + setCentralWidget(table_view_); + + // Assign the selected table below to the table_view_. + // Let's use the Club Competitions table for the purposes of this example. + ehm_dal::Database db; + table_view_->setModel(db.table(ehm_dal::tables::TableIndex::CLUB_COMPETITIONS).get()); +} + +/* MAIN WINDOW CONSTRUCTOR + * Nothing to see here. + * The various class members will be auto-deleted as part of Qt's garbage collection. Hence no explicit `delete` calls here. + */ +MainWindow::~MainWindow() {} + +/* FUNCTION: EXPORT SPREADSHEET (all columns) + * Generates a new spreadsheet and populates it with the entirety of the Club Competitions data table. + */ +void MainWindow::exportSpreadsheet() +{ + ehm_dal::spreadsheet::Spreadsheet spreadsheet; + + // Let's use the Club Competitions table + ehm_dal::Database db; + auto table{db.table(ehm_dal::tables::TableIndex::CLUB_COMPETITIONS)}; + + // Show a window dialog prompting the user to select the destination for saving the spreadsheet. + // The `table->attributes()->name()` function takes the name of the table and this is then used as default file name for saving. + // Abort if the user clicks `Cancel` (`showSaveDialog()` will return false if cancelled). + // By adding the data to the spreadsheet only after this function, we avoid wasting time populating the spreadsheet with data + // in the event the user has clicked `Cancel`. + if (!spreadsheet.showSaveDialog(QString("%1.xlsx").arg(table->attributes()->name()))) + return; + + // Add a header row to the spreadsheet based on the table's column names + spreadsheet.addHeader(table.get()); + // Add the entirety of the table data to the spreadsheet + spreadsheet.add(table.get()); + + // Save the spreadsheet data to the file. + if (spreadsheet.write()) + qInfo() << "Spreadsheet successfully saved"; + else + qInfo() << "Unable to save the spreadsheet"; +} + +/* FUNCTION: EXPORT SPREADSHEET (selected columns) + * Generates a new spreadsheet and populates it with just a few columns of data of the Club Competitions data table. + */ +void MainWindow::exportSpreadsheetSelectedColumns() +{ + // This is a very similar function to `exportSpreadsheet()` but with the key difference that here we generate a list of + // columns we want to export. This will result in just those selected columns being exported. + + ehm_dal::spreadsheet::Spreadsheet spreadsheet; + + ehm_dal::Database db; + auto table{db.table(ehm_dal::tables::TableIndex::CLUB_COMPETITIONS)}; + + if (!spreadsheet.showSaveDialog(QString("%1.xlsx").arg(table->attributes()->name()))) + return; + + // Here's where the function is different to `exportSpreadsheet()` above: + // First we create a list of what columns we want to export. + // We then pass this list to the same `addHeader()` and `add()` functions as before. + std::vector column_list{ehm_dal::column_data::ClubCompetitionColumns::Name, + ehm_dal::column_data::ClubCompetitionColumns::Nation, + ehm_dal::column_data::ClubCompetitionColumns::Classification, + ehm_dal::column_data::ClubCompetitionColumns::Reputation}; + + spreadsheet.addHeader(table.get(), column_list); + spreadsheet.add(table.get(), column_list); + + if (spreadsheet.write()) + qInfo() << "Spreadsheet successfully saved"; + else + qInfo() << "Unable to save the spreadsheet"; +} + +/* FUNCTION: OPEN DATABASE + * This function uses the DatabaseFileManager to display a dialog window allowing the user to select a database file to open. + * The DatabaseFileManager will then proceed to open and process the database file. + */ +void MainWindow::openDatabase() +{ + ehm_dal::file_io::DatabaseFileManager db_file_manager; + + // Display the file selection dialog window and process the selected database file + if (db_file_manager.open(this)) + qInfo() << "Database successfully opened"; + else + qInfo() << "Unable to open the database"; +} + +/* FUNCTION: OPEN SAVED GAME + * This function uses the SavedGameFileManager to display a dialog window allowing the user to select a saved game file to open. + * The SavedGameFileManager will then proceed to open and process the saved game file. + */ +void MainWindow::openSavedGame() +{ + ehm_dal::file_io::SavedGameFileManager sg_file_manager; + + // Display the file selection dialog window and process the selected saved game file + if (sg_file_manager.open(this)) + qInfo() << "Saved game successfully opened"; + else + qInfo() << "Unable to open the saved game"; +} diff --git a/examples/01-01_Simple_Table_View/mainwindow.h b/examples/01-01_Simple_Table_View/mainwindow.h new file mode 100644 index 0000000..762ecec --- /dev/null +++ b/examples/01-01_Simple_Table_View/mainwindow.h @@ -0,0 +1,45 @@ +#pragma once + +/* *************************************** + * EHM Data Abstraction Layer + * by Archibalduk + * + * https://github.com/archibalduk/EHM_DAL + * + * Example 01.01 - Simple Table View + * + * ***************************************/ + +#include +#include +#include +#include + +/// The purpose of this example is to show how easily a database table can be displayed using a QTableView widget. +/// This example additionally shows how table data can be exported to a csv/xlsx spreadsheet file. +class MainWindow : public QMainWindow +{ + Q_OBJECT + +public: + MainWindow(QWidget *parent = nullptr); + ~MainWindow(); + +private: + // Menu + QMenuBar *menu_bar_; + QAction *open_database_action_; + QAction *open_saved_game_action_; + QAction *spreadsheet_export_all_columns_action_; + QAction *spreadsheet_export_selected_columns_action_; + + // Table + QTableView *table_view_; + +private slots: + // Slot functions for connecting to signal functions + void exportSpreadsheet(); + void exportSpreadsheetSelectedColumns(); + void openDatabase(); + void openSavedGame(); +};