From 45a3ade1cb3521e0406de28859d3d4ca7a298b00 Mon Sep 17 00:00:00 2001 From: larteyoh Date: Mon, 10 Jul 2023 05:42:17 -0400 Subject: [PATCH] add favorites (wishlist) feature --- qml/components/CatalogGrid.qml | 8 +- qml/components/ProductDialog.qml | 8 +- qml/components/TagField.qml | 2 +- src/core/category.hpp | 38 ++-- src/core/image.hpp | 3 +- src/core/protocol/p2p/serializer.cpp | 2 +- src/core/seller.cpp | 4 +- src/core/tools/base64.cpp | 72 +++++++ src/core/tools/base64.hpp | 4 + src/core/user.cpp | 170 ++++++++-------- src/core/user.hpp | 12 +- src/gui/backend.cpp | 281 +++++++++++++-------------- src/gui/backend.hpp | 9 +- src/gui/user_controller.cpp | 17 ++ src/gui/user_controller.hpp | 6 +- tests/request/request | Bin 29856 -> 29856 bytes tests/request/request.cpp | 2 +- tests/request/request.sh | 2 +- 18 files changed, 367 insertions(+), 273 deletions(-) diff --git a/qml/components/CatalogGrid.qml b/qml/components/CatalogGrid.qml index 3fe6464..2bb3603 100644 --- a/qml/components/CatalogGrid.qml +++ b/qml/components/CatalogGrid.qml @@ -141,9 +141,9 @@ GridView { anchors.rightMargin: 5//10 anchors.top: parent.top anchors.topMargin: 5//10 - property bool disabled: true + property bool disabled: !User.hasFavorited(modelData.key) icon.source: "qrc:/assets/images/heart.png" - icon.color: NeroshopComponents.Style.disabledColor//"#ffffff" + icon.color: heartIconButton.disabled ? "#808080" : "#e05d5d" icon.height: 24; icon.width: 24 background: Rectangle { color: "transparent" @@ -154,10 +154,12 @@ GridView { if(disabled) { disabled = false icon.color = "#e05d5d" + User.addToFavorites(modelData.key) } else { disabled = true - icon.color = NeroshopComponents.Style.disabledColor + icon.color = "#808080" + User.removeFromFavorites(modelData.key) } } MouseArea { diff --git a/qml/components/ProductDialog.qml b/qml/components/ProductDialog.qml index 5eb09e7..b6efb91 100644 --- a/qml/components/ProductDialog.qml +++ b/qml/components/ProductDialog.qml @@ -310,7 +310,7 @@ Popup { currentIndex = find("Miscellaneous") } function reset() { - let subcategories = Backend.getSubCategoryList(Backend.getCategoryIdByName(productCategoryBox.currentText)) + let subcategories = Backend.getSubCategoryList(Backend.getCategoryIdByName(productCategoryBox.currentText), true) addSubCategoryButton.visible = (subcategories.length > 0) subCategoryRepeater.model = 0 // reset } @@ -337,7 +337,7 @@ Popup { horizontalAlignment: Text.AlignHCenter } onClicked: { - let subcategories = Backend.getSubCategoryList(Backend.getCategoryIdByName(productCategoryBox.currentText)) + let subcategories = Backend.getSubCategoryList(Backend.getCategoryIdByName(productCategoryBox.currentText), true) if(subCategoryRepeater.count == 1) { console.log("Cannot add no more than 1 subcategories") return @@ -358,7 +358,7 @@ Popup { function getSubCategoryStringList() { let subCategoryStringList = [] - let subcategories = Backend.getSubCategoryList(Backend.getCategoryIdByName(productCategoryBox.currentText)) + let subcategories = Backend.getSubCategoryList(Backend.getCategoryIdByName(productCategoryBox.currentText), true) for(let i = 0; i < subcategories.length; i++) { subCategoryStringList[i] = subcategories[i].name//console.log(parent.parent.parent.categoryStringList[i])//console.log(categories[i].name) } @@ -785,6 +785,8 @@ Popup { productConditionBox.currentText, productLocationBox.currentText ) + // Save product thumbnail + Backend.saveProductThumbnail(productImages[0].source, listing_key) // Save product image(s) to datastore folder for (let i = 0; i < productImages.length; i++) { Backend.saveProductImage(productImages[i].source, listing_key) diff --git a/qml/components/TagField.qml b/qml/components/TagField.qml index aa51faa..f4d1da7 100644 --- a/qml/components/TagField.qml +++ b/qml/components/TagField.qml @@ -104,7 +104,7 @@ Item { Layout.preferredHeight: 50//height: 30 placeholderText: "Add tags (comma-separated)" - onAccepted: { + onEditingFinished: { if (tagInput.text.trim() !== "") { let tags = tagInput.text.split(",") tags = tags.map(function(tag) { return tag.trim() }) diff --git a/src/core/category.hpp b/src/core/category.hpp index b682957..61fd308 100644 --- a/src/core/category.hpp +++ b/src/core/category.hpp @@ -44,26 +44,40 @@ const std::vector predefined_categories = { { 21, "Real Estate, Property & Housing", "", ""}, { 22, "Luggage & Travel", "", ""}, { 23, "Business, Industrial & Scientific", "", ""}, - { 24, "Illegal", "Banned and/or prohibited items", ""}, + { 24, "Weapons & Firearms", "Firearms;Ammunition;Explosives", ""}, + { 25, "Illegal", "Banned and/or prohibited items", ""}, }; const std::vector predefined_subcategories = { // For each category is a subcategory (for example, the books, movies, and music categories can have the digital goods subcategory if they are digital rather than physical) // Digital Goods - { static_cast(predefined_categories.size() + 1), predefined_categories[5].name, predefined_categories[5].description, predefined_categories[5].thumbnail, 7 }, - { 26, predefined_categories[5].name, predefined_categories[5].description, predefined_categories[5].thumbnail, 8 }, - { 27, predefined_categories[5].name, predefined_categories[5].description, predefined_categories[5].thumbnail, 9 }, + { static_cast(predefined_categories.size() + 0), predefined_categories[5].name, predefined_categories[5].description, predefined_categories[5].thumbnail, 7 }, + { static_cast(predefined_categories.size() + 1), predefined_categories[5].name, predefined_categories[5].description, predefined_categories[5].thumbnail, 8 }, + { static_cast(predefined_categories.size() + 2), predefined_categories[5].name, predefined_categories[5].description, predefined_categories[5].thumbnail, 9 }, // Illegal - { 28, predefined_categories[24].name, predefined_categories[24].description, predefined_categories[24].thumbnail, 0 }, - { 29, predefined_categories[24].name, predefined_categories[24].description, predefined_categories[24].thumbnail, 5 }, - { 30, predefined_categories[24].name, predefined_categories[24].description, predefined_categories[24].thumbnail, 6 }, - { 31, predefined_categories[24].name, predefined_categories[24].description, predefined_categories[24].thumbnail, 7 }, - { 32, predefined_categories[24].name, predefined_categories[24].description, predefined_categories[24].thumbnail, 8 }, - { 33, predefined_categories[24].name, predefined_categories[24].description, predefined_categories[24].thumbnail, 23 }, - { 34, predefined_categories[24].name, predefined_categories[24].description, predefined_categories[24].thumbnail, 19 }, + { static_cast(predefined_categories.size() + 3), predefined_categories[25].name, predefined_categories[25].description, predefined_categories[25].thumbnail, 0 }, + { static_cast(predefined_categories.size() + 4), predefined_categories[25].name, predefined_categories[25].description, predefined_categories[25].thumbnail, 5 }, + { static_cast(predefined_categories.size() + 5), predefined_categories[25].name, predefined_categories[25].description, predefined_categories[25].thumbnail, 6 }, + { static_cast(predefined_categories.size() + 6), predefined_categories[25].name, predefined_categories[25].description, predefined_categories[25].thumbnail, 7 }, + { static_cast(predefined_categories.size() + 7), predefined_categories[25].name, predefined_categories[25].description, predefined_categories[25].thumbnail, 8 }, + { static_cast(predefined_categories.size() + 8), predefined_categories[25].name, predefined_categories[25].description, predefined_categories[25].thumbnail, 19 }, + { static_cast(predefined_categories.size() + 9), predefined_categories[25].name, predefined_categories[25].description, predefined_categories[25].thumbnail, 23 }, + { static_cast(predefined_categories.size() + 10), predefined_categories[25].name, predefined_categories[25].description, predefined_categories[25].thumbnail, 24 }, - //{ 0, , }, + // All categories + { static_cast(predefined_categories.size() + 11), predefined_categories[0].name, predefined_categories[0].description, predefined_categories[0].thumbnail, 25 }, + { static_cast(predefined_categories.size() + 12), predefined_categories[5].name, predefined_categories[5].description, predefined_categories[5].thumbnail, 25 }, + { static_cast(predefined_categories.size() + 13), predefined_categories[6].name, predefined_categories[6].description, predefined_categories[6].thumbnail, 25 }, + { static_cast(predefined_categories.size() + 14), predefined_categories[7].name, predefined_categories[7].description, predefined_categories[7].thumbnail, 25 }, + { static_cast(predefined_categories.size() + 15), predefined_categories[8].name, predefined_categories[8].description, predefined_categories[8].thumbnail, 25 }, + { static_cast(predefined_categories.size() + 16), predefined_categories[19].name, predefined_categories[19].description, predefined_categories[19].thumbnail, 25 }, + { static_cast(predefined_categories.size() + 17), predefined_categories[23].name, predefined_categories[23].description, predefined_categories[23].thumbnail, 25 }, + { static_cast(predefined_categories.size() + 18), predefined_categories[24].name, predefined_categories[24].description, predefined_categories[24].thumbnail, 25 }, + + { static_cast(predefined_categories.size() + 19), predefined_categories[7].name, predefined_categories[7].description, predefined_categories[7].thumbnail, 5 }, + { static_cast(predefined_categories.size() + 20), predefined_categories[8].name, predefined_categories[8].description, predefined_categories[8].thumbnail, 5 }, + { static_cast(predefined_categories.size() + 21), predefined_categories[9].name, predefined_categories[9].description, predefined_categories[9].thumbnail, 5 }, }; //----------------------------------------------------------------------------- diff --git a/src/core/image.hpp b/src/core/image.hpp index bb7058b..ee16203 100644 --- a/src/core/image.hpp +++ b/src/core/image.hpp @@ -8,7 +8,8 @@ namespace neroshop { struct Image { std::string name; // base name + ext size_t size; // bytes - unsigned int id = 0; // The specific order in which images are displayed//bool thumbnail; + unsigned int id = 0; // The specific order in which images are displayed + std::string source; // never stored in DHT }; } diff --git a/src/core/protocol/p2p/serializer.cpp b/src/core/protocol/p2p/serializer.cpp index bc16142..a158eaf 100644 --- a/src/core/protocol/p2p/serializer.cpp +++ b/src/core/protocol/p2p/serializer.cpp @@ -137,7 +137,7 @@ std::pair*/> neroshop::Serializer image_obj["size"] = image.size; image_obj["id"] = image.id; bool is_thumbnail = ((images.size() == 1) || (image.id == 0)); - if(is_thumbnail) { // TODO: store only thumbnail images in DHT + if(is_thumbnail) { std::cout << image.name << " \033[1;35mwill be used as thumbnail\033[0m\n"; } diff --git a/src/core/seller.cpp b/src/core/seller.cpp index ecbdd0a..fb22367 100644 --- a/src/core/seller.cpp +++ b/src/core/seller.cpp @@ -668,10 +668,10 @@ neroshop::User * neroshop::Seller::on_login(const neroshop::Wallet& wallet) { // dynamic_cast(user)->set_account_type(UserAccountType::Seller); //------------------------------- /*// load orders - dynamic_cast(user)->load_orders(); + dynamic_cast(user)->load_orders();*/ // load wishlists dynamic_cast(user)->load_favorites(); - // load customer_orders + /*// load customer_orders static_cast(user)->load_customer_orders();*/ // Load cart (into memory) user->get_cart()->load(user->get_id()); diff --git a/src/core/tools/base64.cpp b/src/core/tools/base64.cpp index c629cf7..9bf2e9b 100644 --- a/src/core/tools/base64.cpp +++ b/src/core/tools/base64.cpp @@ -3,6 +3,7 @@ #include #include #include +#include std::string neroshop::base64_encode(const std::string& input) { // Create a BIO object for Base64 encoding @@ -55,6 +56,77 @@ std::string neroshop::base64_decode(const std::string& encoded) { return decoded_data; } +//---------------------------------------------------------------------------- + +std::string neroshop::base64_image_encode(const std::string& image_path) { + // Open the image file + std::ifstream image_file(image_path, std::ios::binary); + if (!image_file.is_open()) { + // Error handling for failed file opening + return ""; + } + + // Get the size of the file + image_file.seekg(0, std::ios::end); + size_t file_size = image_file.tellg(); + image_file.seekg(0, std::ios::beg); + + // Read the image data into a buffer + std::vector buffer(file_size); + image_file.read(buffer.data(), file_size); + + // Create a BIO object for Base64 encoding + BIO* b64 = BIO_new(BIO_f_base64()); + BIO_set_flags(b64, BIO_FLAGS_BASE64_NO_NL); + + // Create a BIO object for memory buffering + BIO* mem = BIO_new(BIO_s_mem()); + BIO_push(b64, mem); + + // Write input data to the BIO for encoding + BIO_write(b64, buffer.data(), static_cast(buffer.size())); + BIO_flush(b64); + + // Read the encoded data from the BIO + BUF_MEM* mem_buf; + BIO_get_mem_ptr(b64, &mem_buf); + + // Copy the encoded data to a string + std::string encoded_data(mem_buf->data, mem_buf->length); + + // Cleanup + BIO_free_all(b64); + + return encoded_data; +} + +std::vector neroshop::base64_image_decode(const std::string& encoded) { + // Create a BIO object for Base64 decoding + BIO* b64 = BIO_new(BIO_f_base64()); + BIO_set_flags(b64, BIO_FLAGS_BASE64_NO_NL); + + // Create a BIO object for memory buffering + BIO* mem = BIO_new_mem_buf(encoded.c_str(), static_cast(encoded.length())); + BIO_push(b64, mem); + + // Determine the size of the decoded data + size_t max_decoded_length = (encoded.length() * 3) / 4; + std::vector decoded_data(max_decoded_length); + + // Decode the data + int decoded_length = BIO_read(b64, decoded_data.data(), static_cast(decoded_data.size())); + + // Resize the vector to the actual decoded length + decoded_data.resize(decoded_length); + + // Cleanup + BIO_free_all(b64); + + return decoded_data; +} + +//---------------------------------------------------------------------------- + /*int main() { std::string input = "Hello, World!"; diff --git a/src/core/tools/base64.hpp b/src/core/tools/base64.hpp index be325d7..d7f2476 100644 --- a/src/core/tools/base64.hpp +++ b/src/core/tools/base64.hpp @@ -2,10 +2,14 @@ #include #include +#include namespace neroshop { extern std::string base64_encode(const std::string& input); extern std::string base64_decode(const std::string& encoded); +// untested +extern std::string base64_image_encode(const std::string& image_path); +extern std::vector base64_image_decode(const std::string& encoded); } diff --git a/src/core/user.cpp b/src/core/user.cpp index 77f5b6e..5a27ea3 100644 --- a/src/core/user.cpp +++ b/src/core/user.cpp @@ -11,7 +11,7 @@ #include //////////////////// -neroshop::User::User() : id(""), logged(false), account_type(UserAccountType::Guest), cart(nullptr), order_list({}), favorites_list({}) { +neroshop::User::User() : id(""), logged(false), account_type(UserAccountType::Guest), cart(nullptr), order_list({}), favorites({}) { cart = std::unique_ptr(new Cart()); } //////////////////// @@ -24,7 +24,7 @@ neroshop::User::~User() // clear orders order_list.clear(); // this should reset (delete) all orders // clear favorites - favorites_list.clear(); // this should reset (delete) all favorites + favorites.clear(); // this should reset (delete) all favorites #ifdef NEROSHOP_DEBUG std::cout << "user deleted\n"; #endif @@ -239,18 +239,10 @@ void neroshop::User::add_to_cart(const std::string& product_id, int quantity) { cart->add(this->id, product_id, quantity); } //////////////////// -void neroshop::User::add_to_cart(const neroshop::Product& item, int quantity) { - add_to_cart(item.get_id(), quantity); -} -//////////////////// void neroshop::User::remove_from_cart(const std::string& product_id, int quantity) { cart->remove(this->id, product_id, quantity); } //////////////////// -void neroshop::User::remove_from_cart(const neroshop::Product& item, int quantity) { - remove_from_cart(item.get_id(), quantity); -} -//////////////////// void neroshop::User::clear_cart() { cart->empty(); } @@ -331,78 +323,92 @@ void neroshop::User::load_orders() { //////////////////// // favorite-or-wishlist-related stuff //////////////////// -void neroshop::User::add_to_favorites(const std::string& product_id) { -#if defined(NEROSHOP_USE_POSTGRESQL) +void neroshop::User::add_to_favorites(const std::string& listing_key) { + db::Sqlite3 * database = neroshop::get_database(); + // check if item is already in favorites so that we do not add the same item more than once - std::string item_name = database->get_text_params("SELECT name FROM item WHERE id = $1", { product_id }); - bool favorited = (database->get_text_params("SELECT EXISTS(SELECT product_ids FROM favorites WHERE $1 = ANY(product_ids) AND user_id = $2)", { product_id, std::to_string(this->id) }) == "t") ? true : false; - if(favorited) { neroshop::print("\"" + item_name + "\" is already in your favorites", 2); return; /* exit function */} + bool favorited = database->get_integer_params("SELECT EXISTS(SELECT listing_key FROM favorites WHERE listing_key = ?1 AND user_id = ?2)", { listing_key, this->id }); + if(favorited) { neroshop::print("\"" + listing_key + "\" is already in your favorites", 2); return; } // add item to favorites - database->execute_params("UPDATE favorites SET product_ids = array_append(product_ids, $1::integer) WHERE user_id = $2", { product_id, std::to_string(this->id) }); - // store in vector as well - favorites_list.push_back(std::make_shared(product_id));//(std::unique_ptr(new neroshop::Product(product_id))); - neroshop::print("\"" + item_name + "\" has been added to your favorites", 3);//if(std::find(favorites_list.begin(), favorites_list.end(), product_id) == favorites_list.end()) { favorites_list.push_back(product_id); neroshop::print("\"" + item_name + "\" has been added to your favorites", 3); }// this works for a favorite_list that stores integers (product_ids) rather than the item object itself -#endif -} -//////////////////// -void neroshop::User::add_to_favorites(const neroshop::Product& item) { - add_to_favorites(item.get_id()); + int rescode = database->execute_params("INSERT INTO favorites (user_id, listing_key) VALUES (?1, ?2);", { this->id, listing_key }); + if(rescode != SQLITE_OK) { + neroshop::print("failed to add item to favorites", 1); + return; + } + // store in memory as well + favorites.push_back(listing_key); + if(std::find(favorites.begin(), favorites.end(), listing_key) != favorites.end()) { + neroshop::print("\"" + listing_key + "\" has been added to your favorites", 3); + } } //////////////////// -void neroshop::User::remove_from_favorites(const std::string& product_id) { -#if defined(NEROSHOP_USE_POSTGRESQL) +void neroshop::User::remove_from_favorites(const std::string& listing_key) { + db::Sqlite3 * database = neroshop::get_database(); + // check if item has already been removed from favorites so that we don't have to remove it more than once - std::string item_name = database->get_text_params("SELECT name FROM item WHERE id = $1", { product_id }); - bool favorited = (database->get_text_params("SELECT EXISTS(SELECT product_ids FROM favorites WHERE $1 = ANY(product_ids) AND user_id = $2)", { product_id, std::to_string(this->id) }) == "t") ? true : false; - if(!favorited) return; // exit function ////{ neroshop::print("\"" + item_name + "\" was not found in your favorites list", 2); return; } - // remove item from favorites - database->execute_params("UPDATE favorites SET product_ids = array_remove(product_ids, $1::integer) WHERE user_id = $2", { product_id, std::to_string(this->id) }); + bool favorited = database->get_integer_params("SELECT EXISTS(SELECT listing_key FROM favorites WHERE listing_key = ?1 AND user_id = ?2)", { listing_key, this->id }); + if(!favorited) { + auto it = std::find(favorites.begin(), favorites.end(), listing_key); + if (it != favorites.end()) { favorites.erase(it); } // remove from vector if found in-memory, but not found in database + neroshop::print("\"" + listing_key + "\" is not in your favorites", 2); + return; + } + // remove item from favorites (database) + int rescode = database->execute_params("DELETE FROM favorites WHERE listing_key = ?1 AND user_id = ?2;", { listing_key, this->id }); + if (rescode != SQLITE_OK) { + neroshop::print("failed to remove item from favorites", 1); + return; + } // remove from vector as well - for(const auto & favorites : favorites_list) { - if(favorites->get_id() != product_id) continue; // skip items whose ids do the match the product_id to be deleted - auto it = std::find(favorites_list.begin(), favorites_list.end(), favorites); - int item_index = it - favorites_list.begin();//std::cout << "favorites_list item index: " << item_index << std::endl; - favorites_list.erase(favorites_list.begin() + item_index); - if(std::find(favorites_list.begin(), favorites_list.end(), favorites) == favorites_list.end()) neroshop::print("\"" + item_name + "\" has been removed from your favorites", 1); // confirm that item has been removed from favorites_list - } -#endif -} -//////////////////// -void neroshop::User::remove_from_favorites(const neroshop::Product& item) { - remove_from_favorites(item.get_id()); + auto it = std::find(favorites.begin(), favorites.end(), listing_key); + if (it != favorites.end()) { + favorites.erase(it); + if(std::find(favorites.begin(), favorites.end(), listing_key) == favorites.end()) neroshop::print("\"" + listing_key + "\" has been removed from your favorites", 1); // confirm that item has been removed from favorites + } } //////////////////// void neroshop::User::clear_favorites() { -#if defined(NEROSHOP_USE_POSTGRESQL) - // first check if array is empty - int is_empty = database->get_integer_params("SELECT COUNT(*) FROM favorites WHERE product_ids = '{}' AND user_id = $1", { std::to_string(this->id) }); - if(is_empty) return; // array is empty so that means there is nothing to delete, exit function + db::Sqlite3 * database = neroshop::get_database(); + + // first check if favorites (database table) is empty + int favorites_count = database->get_integer_params("SELECT COUNT(*) FROM favorites WHERE user_id = ?1", { this->id }); + if(favorites_count <= 0) return; // table is empty so that means there is nothing to delete, exit function // clear all items from favorites - database->execute_params("UPDATE favorites SET product_ids = '{}' WHERE user_id = $1", { std::to_string(this->id) }); + database->execute_params("DELETE FROM favorites WHERE user_id = ?1", { this->id }); // clear favorites from vector as well - favorites_list.clear(); - if(favorites_list.empty()) neroshop::print("your favorites have been cleared");// confirm that favorites_list has been cleared -#endif + favorites.clear(); + if(favorites.empty()) neroshop::print("your favorites have been cleared"); // confirm that favorites has been cleared } //////////////////// void neroshop::User::load_favorites() { -#if defined(NEROSHOP_USE_POSTGRESQL) - std::string command = "SELECT unnest(product_ids) FROM favorites WHERE user_id = $1"; - std::vector param_values = { std::to_string(this->id).c_str() }; - PGresult * result = PQexecParams(database->get_handle(), command.c_str(), 1, nullptr, param_values.data(), nullptr, nullptr, 0); - int rows = PQntuples(result);//if(rows < 1) { PQclear(result); return; } - if (PQresultStatus(result) != PGRES_TUPLES_OK) { - neroshop::print("User::load_favorites(): Your favorites list is empty", 2); - PQclear(result);//exit(1); - return; // exit so that we don't double free "result" + favorites.clear(); + db::Sqlite3 * database = neroshop::get_database(); + std::string command = "SELECT DISTINCT listing_key FROM favorites WHERE user_id = ?1;"; + sqlite3_stmt * stmt = nullptr; + // Prepare (compile) statement + if(sqlite3_prepare_v2(database->get_handle(), command.c_str(), -1, &stmt, nullptr) != SQLITE_OK) { + neroshop::print("sqlite3_prepare_v2: " + std::string(sqlite3_errmsg(database->get_handle())), 1); + return; } - for(int i = 0; i < rows; i++) { - int product_id = std::stoi(PQgetvalue(result, i, 0)); - favorites_list.push_back(std::make_shared(product_id));//(std::unique_ptr(new neroshop::Product(product_id))); // store favorited_items for later use - neroshop::print("Favorited item (id: " + product_id + ") has been loaded"); + // Bind this->id to first argument + if(sqlite3_bind_text(stmt, 1, this->id.c_str(), this->id.length(), SQLITE_STATIC) != SQLITE_OK) { + neroshop::print("sqlite3_bind_text (arg: 1): " + std::string(sqlite3_errmsg(database->get_handle())), 1); + sqlite3_finalize(stmt); + return; + } + + while(sqlite3_step(stmt) == SQLITE_ROW) { + for(int i = 0; i < sqlite3_column_count(stmt); i++) { + std::string listing_key = (sqlite3_column_text(stmt, i) == nullptr) ? "NULL" : reinterpret_cast(sqlite3_column_text(stmt, i)); + if(listing_key == "NULL") continue; // Skip invalid columns + favorites.push_back(listing_key); // store favorited listings for later use + if(std::find(favorites.begin(), favorites.end(), listing_key) != favorites.end()) { + neroshop::print("Favorited item (" + listing_key + ") has been loaded"); + } + } } - PQclear(result); -#endif + + sqlite3_finalize(stmt); } //////////////////// //////////////////// @@ -690,20 +696,16 @@ std::vector neroshop::User::get_order_list() const { } //////////////////// //////////////////// -neroshop::Product * neroshop::User::get_favorite(unsigned int index) const { - if(index > (favorites_list.size() - 1)) throw std::out_of_range("neroshop::User::get_favorites(): attempt to access invalid index"); - return favorites_list[index].get(); +std::string neroshop::User::get_favorite(unsigned int index) const { + if(index > (favorites.size() - 1)) throw std::out_of_range("neroshop::User::get_favorites(): attempt to access invalid index"); + return favorites[index]; } //////////////////// unsigned int neroshop::User::get_favorites_count() const { - return favorites_list.size(); + return favorites.size(); } //////////////////// -std::vector neroshop::User::get_favorites_list() const { - std::vector favorites = {}; - for(const auto & item : favorites_list) {//for(int f = 0; f < favorites_list.size(); f++) { - favorites.push_back(item.get());//(favorites_list[f].get()); - } +std::vector neroshop::User::get_favorites() const { return favorites; } //////////////////// @@ -906,21 +908,13 @@ bool neroshop::User::has_purchased(const std::string& product_id) { // for regis return false; } //////////////////// -bool neroshop::User::has_purchased(const neroshop::Product& item) { - return has_purchased(item.get_id()); -} -//////////////////// -bool neroshop::User::has_favorited(const std::string& product_id) { +bool neroshop::User::has_favorited(const std::string& listing_key_or_id) { // since we loaded the favorites into memory when the app launched, we should be able to access the pre-loaded favorites and any newly added favorites in the current session without performing any database queries/operations - for(const auto & favorites : favorites_list) { - // if any favorites_list items' ids matches "product_id" then return true - if(favorites->get_id() == product_id) return true; + for(const auto & favorite : favorites) { + // if any favorites items' ids matches "product_id" then return true + if(favorite == listing_key_or_id) return true; } - return false;////return (std::find(favorites_list.begin(), favorites_list.end(), product_id) != favorites_list.end()); // this is good for when storing favorites as integers (product_ids) -} -//////////////////// -bool neroshop::User::has_favorited(const neroshop::Product& item) { - return has_favorited(item.get_id()); + return false;////return (std::find(favorites.begin(), favorites.end(), product_id) != favorites.end()); // this is good for when storing favorites as integers (product_ids) } //////////////////// //////////////////// diff --git a/src/core/user.hpp b/src/core/user.hpp index 37f9a09..3e29637 100644 --- a/src/core/user.hpp +++ b/src/core/user.hpp @@ -27,17 +27,13 @@ public: void logout(); // cart-related stuff (50% complete - cart class still needs some more work) void add_to_cart(const std::string& product_id, int quantity = 1); - void add_to_cart(const neroshop::Product& item, int quantity = 1); // use int and NOT unsigned int 'cause unsigned int assumes the arg will never be negative number, but when arg is negative, it converts it to some random positive number void remove_from_cart(const std::string& product_id, int quantity = 1); - void remove_from_cart(const neroshop::Product& item, int quantity = 1); void clear_cart(); // order-related stuff (50% complete - order class still needs some more work) void create_order(const std::string& shipping_address);// const;//void create_order(); // favorite-or-wishlist-related stuff (100% complete) void add_to_favorites(const std::string& product_id); - void add_to_favorites(const neroshop::Product& item); void remove_from_favorites(const std::string& product_id); - void remove_from_favorites(const neroshop::Product& item); void clear_favorites(); // avatar-related stuff (10% complete) void upload_avatar(const std::string& filename); @@ -58,9 +54,9 @@ public: neroshop::Order * get_order(unsigned int index) const; unsigned int get_order_count() const; std::vector get_order_list() const; - neroshop::Product * get_favorite(unsigned int index) const; + std::string get_favorite(unsigned int index) const; unsigned int get_favorites_count() const; - std::vector get_favorites_list() const; + std::vector get_favorites() const; // boolean bool is_guest() const; bool is_buyer() const; @@ -73,9 +69,7 @@ public: bool has_avatar() const; // item-related stuff - boolean bool has_purchased(const std::string& product_id); // checks if an item was previously purchased or not - bool has_purchased(const neroshop::Product& item); // checks if an item was previously purchased or not bool has_favorited(const std::string& product_id); // checks if an item is in a user's favorites or wishlist - bool has_favorited(const neroshop::Product& item); // checks if an item is in a user's favorites or wishlist // callbacks void on_registration(const std::string& name); // on registering an account //virtual User * on_login(const std::string& username);// = 0; // load all data: orders, reputation/ratings, settings // for all users @@ -105,7 +99,7 @@ private: std::string private_key; std::unique_ptr cart; std::vector> order_list; - std::vector> favorites_list; // I get the error "/usr/include/c++/9/bits/stl_uninitialized.h:127:72: error: static assertion failed: result type must be constructible from value type of input range" while trying to use unique_ptr so I'm stuck with a shared_ptr container for now + std::vector favorites; std::string get_private_key() const; }; } diff --git a/src/gui/backend.cpp b/src/gui/backend.cpp index 6b4eda4..9008bc7 100644 --- a/src/gui/backend.cpp +++ b/src/gui/backend.cpp @@ -14,6 +14,13 @@ #include // Note: QProcess is not supported on VxWorks, iOS, tvOS, or watchOS. #include #include +#include +#include +#include +#include +#include +#include +#include #include "../neroshop_config.hpp" #include "../core/version.hpp" @@ -34,7 +41,6 @@ #include "../core/category.hpp" #include "../core/tools/regex.hpp" #include "../core/crypto/rsa.hpp" -#include "../core/protocol/p2p/mapper.hpp" #include #include @@ -62,6 +68,23 @@ void neroshop::Backend::copyTextToClipboard(const QString& text) { } //---------------------------------------------------------------- //---------------------------------------------------------------- +QString neroshop::Backend::imageToBase64(const QImage& image) { + QByteArray byteArray; + QBuffer buffer(&byteArray); + buffer.open(QIODevice::WriteOnly); + image.save(&buffer, "PNG"); // You can choose a different format if needed + return QString::fromLatin1(byteArray.toBase64().data()); +} +//---------------------------------------------------------------- +QImage neroshop::Backend::base64ToImage(const QString& base64Data) { + QByteArray byteArray = QByteArray::fromBase64(base64Data.toLatin1()); + QImageReader reader(byteArray); + reader.setAutoTransform(true); + const QImage image = reader.read(); + return image; +} +//---------------------------------------------------------------- +//---------------------------------------------------------------- QStringList neroshop::Backend::getCurrencyList() const { QStringList currency_list; @@ -95,8 +118,7 @@ bool neroshop::Backend::isSupportedCurrency(const QString& currency) const { void neroshop::Backend::initializeDatabase() { db::Sqlite3 * database = neroshop::get_database(); database->execute("BEGIN;"); - //------------------------- - // Todo: Make monero_address the primary key and remove id. Also, replace all foreign key references from id to monero_address + // table users if(!database->table_exists("users")) { database->execute("CREATE TABLE users(name TEXT, monero_address TEXT NOT NULL PRIMARY KEY"//, UNIQUE" @@ -107,61 +129,37 @@ void neroshop::Backend::initializeDatabase() { // Notes: Display names are optional which means they can be an empty string but making the "name" column UNIQUE will not allow empty strings on multiple names ////database->execute("CREATE UNIQUE INDEX index_public_keys ON users (public_key);"); // This is commented out to allow multiple users to use the same public key, in the case of a user having two neroshop accounts? } - // products (represents both items and services) - if(!database->table_exists("products")) { - database->execute("CREATE TABLE products(uuid TEXT NOT NULL PRIMARY KEY);"); - database->execute("ALTER TABLE products ADD COLUMN name TEXT;"); - database->execute("ALTER TABLE products ADD COLUMN description TEXT;"); - //database->execute("ALTER TABLE products ADD COLUMN price REAL");// This should be the manufacturer's original price (won't be used though) // unit_price or price_per_unit - database->execute("ALTER TABLE products ADD COLUMN weight REAL;"); // kg // TODO: add weight to attributes - database->execute("ALTER TABLE products ADD COLUMN attributes TEXT;"); // attribute options format: "Color:Red,Green,Blue;Size:XS,S,M,L,XL"// Can be a number(e.g 16) or a text(l x w x h) - database->execute("ALTER TABLE products ADD COLUMN code TEXT;"); // product_code can be either upc (universal product code) or a custom sku - database->execute("ALTER TABLE products ADD COLUMN category_id INTEGER REFERENCES categories(id);"); - //database->execute("ALTER TABLE products ADD COLUMN subcategory_id INTEGER REFERENCES categories(id);"); - //database->execute("ALTER TABLE products ADD COLUMN ?col ?datatype;"); - //database->execute("CREATE UNIQUE INDEX ?index ON products (?col);"); - // the seller determines the final product price, the product condition and whether the product will have a discount or not - // Note: UPC codes can be totally different for the different variations(color, etc.) of the same product + + // mappings + if(!database->table_exists("mappings")) { + database->execute("CREATE VIRTUAL TABLE mappings USING fts5(search_term, key, content, tokenize='porter unicode61');"); } - // listings - if(!database->table_exists("listings")) { - database->execute("CREATE TABLE listings(uuid TEXT NOT NULL PRIMARY KEY, " - "product_id TEXT REFERENCES products(uuid) ON DELETE CASCADE, " // ON DELETE CASCADE keeps the parent table from being deleted until all child rows that references the parent are deleted first - "seller_id TEXT REFERENCES users(monero_address)" // alternative names: "store_id" + + // favorites (wishlists) + if(!database->table_exists("favorites")) { + database->execute("CREATE TABLE favorites(" + "user_id TEXT REFERENCES users(monero_address), " + "listing_key TEXT, " + "UNIQUE(user_id, listing_key)" ");"); - database->execute("ALTER TABLE listings ADD COLUMN quantity INTEGER;"); // stock available - database->execute("ALTER TABLE listings ADD COLUMN price REAL;"); // this is the final price of a product or list/sales price decided by the seller - database->execute("ALTER TABLE listings ADD COLUMN currency TEXT;"); // the fiat currency the seller is selling the item in - //database->execute("ALTER TABLE listings ADD COLUMN discount numeric(20,12);"); // alternative names: "seller_discount", or "discount_price" - //database->execute("ALTER TABLE listings ADD COLUMN ?col ?datatype;"); // discount_times_can_use - number of times the discount can be used - //database->execute("ALTER TABLE listings ADD COLUMN ?col ?datatype;"); // discounted_items_qty - number of items that the discount will apply to - //database->execute("ALTER TABLE listings ADD COLUMN ?col ?datatype;"); // discount_expiry - date and time that the discount expires (will be in UTC format) - //database->execute("ALTER TABLE listings ADD COLUMN ?col ?datatype;"); - database->execute("ALTER TABLE listings ADD COLUMN condition TEXT;"); // item condition - database->execute("ALTER TABLE listings ADD COLUMN location TEXT;"); - //database->execute("ALTER TABLE listings ADD COLUMN last_updated ?datatype;"); - database->execute("ALTER TABLE listings ADD COLUMN date TEXT DEFAULT CURRENT_TIMESTAMP;"); // date when first listed // will use ISO8601 string format as follows: YYYY-MM-DD HH:MM:SS.SSS - //database->execute(""); - // For most recent listings: "SELECT * FROM listings ORDER BY date DESC;" - } + } + // cart if(!database->table_exists("cart")) { - // local cart - for a single cart containing a list of product_ids - // public cart - copied to all peers' databases database->execute("CREATE TABLE cart(uuid TEXT NOT NULL PRIMARY KEY, " "user_id TEXT REFERENCES users(monero_address) ON DELETE CASCADE" ");"); - // cart_items (public cart) + // cart_items database->execute("CREATE TABLE cart_item(id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, " "cart_id TEXT REFERENCES cart(uuid) ON DELETE CASCADE" ");"); database->execute("ALTER TABLE cart_item ADD COLUMN product_id TEXT REFERENCES products(uuid);"); database->execute("ALTER TABLE cart_item ADD COLUMN quantity INTEGER;"); database->execute("ALTER TABLE cart_item ADD COLUMN seller_id TEXT REFERENCES users(monero_address);"); // for a multi-vendor cart, specifying the seller_id is important! - //database->execute("ALTER TABLE cart_item ADD COLUMN item_price numeric;"); // sales_price will be used for the final pricing rather than the retail_price //database->execute("ALTER TABLE cart_item ADD COLUMN item_weight REAL;");//database->execute("CREATE TABLE cart_item(id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, cart_id TEXT REFERENCES cart(id), product_id TEXT REFERENCES products(id), item_qty INTEGER, item_price NUMERIC, item_weight REAL);"); database->execute("CREATE UNIQUE INDEX index_cart_item ON cart_item (cart_id, product_id);"); // cart_id and product_id duo MUST be unqiue for each row } + // orders (purchase_orders) if(!database->table_exists("orders")) { // TODO: rename to order_requests or nah? database->execute("CREATE TABLE orders(uuid TEXT NOT NULL PRIMARY KEY);");//database->execute("ALTER TABLE orders ADD COLUMN ?col ?datatype;"); @@ -191,85 +189,6 @@ void neroshop::Backend::initializeDatabase() { //database->execute("ALTER TABLE order_item ADD COLUMN unit_price ?datatype;"); //database->execute("ALTER TABLE order_item ADD COLUMN ?col ?datatype;"); } - // ratings - product_ratings, seller_ratings - if(!database->table_exists("seller_ratings")) { - database->execute("CREATE TABLE seller_ratings(id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, " - "seller_id TEXT REFERENCES users(monero_address) ON DELETE CASCADE, " - "score INTEGER, " - "user_id TEXT REFERENCES users(monero_address), " - "comments TEXT, signature TEXT" - ");"); - } - if(!database->table_exists("product_ratings")) { - database->execute("CREATE TABLE product_ratings(id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, " - "product_id TEXT REFERENCES products(uuid) ON DELETE CASCADE, " - "stars INTEGER, " - "user_id TEXT REFERENCES users(monero_address), " - "comments TEXT, signature TEXT" - ");"); - } - // images - if(!database->table_exists("images")) { // TODO: rename to product_images? - database->execute("CREATE TABLE images(id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, " - "product_id TEXT REFERENCES products(uuid) ON DELETE CASCADE, " - "name TEXT, data BLOB" - ");"); - } - // favorites (wishlists) - if(!database->table_exists("favorites")) { - //database->execute("CREATE TABLE ?tbl(id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT);"); - //database->execute("ALTER TABLE ?tbl ADD COLUMN user_id TEXT REFERENCES users(monero_address);"); - //database->execute("ALTER TABLE ?tbl ADD COLUMN product_ids integer[];"); - //database->execute("ALTER TABLE ?tbl ADD COLUMN ?col ?datatype;"); - } - // categories, subcategories - // Note: Products can fall under one category and multiple subcategories - if(!database->table_exists("categories")) { // TODO: rename to product_categories? - database->execute("CREATE TABLE categories(id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT);"); - database->execute("ALTER TABLE categories ADD COLUMN name TEXT;"); - database->execute("ALTER TABLE categories ADD COLUMN description TEXT;"); // alternative names - database->execute("ALTER TABLE categories ADD COLUMN thumbnail TEXT;"); - } - if(!database->table_exists("subcategories")) { // TODO: rename to product_subcategories? - database->execute("CREATE TABLE subcategories(id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT);"); - database->execute("ALTER TABLE subcategories ADD COLUMN name TEXT;"); - database->execute("ALTER TABLE subcategories ADD COLUMN category_id INTEGER REFERENCES categories(id);"); - database->execute("ALTER TABLE subcategories ADD COLUMN description TEXT;"); - // categories types - int category_id = 0; - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Food & Beverages', 'Grocery', '') RETURNING id;"); - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Electronics', '', '') RETURNING id;"); - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Home, Furniture & Appliances', 'Domestic Goods;Furniture;Home Appliances', '') RETURNING id;"); - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Patio & Garden', '', '') RETURNING id;"); - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Digital Goods', '', '') RETURNING id;"); - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Services', 'Non-product services, Freelancing, etc.', '') RETURNING id;"); - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Books', '', '') RETURNING id;"); - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Movies & TV Shows', '', '') RETURNING id;"); - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Music & Vinyl', 'Musical instruments', '') RETURNING id;"); - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Apparel', 'Clothing, Shoes and Accessories; Jewelry and Watches; Fashion', '') RETURNING id;"); - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Pets', 'Domesticated Animals', '') RETURNING id;"); - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Toys & Games', '', '') RETURNING id;"); - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Baby', '', '') RETURNING id;"); - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Arts, Crafts, Sewing & Party Supplies', 'DIY & Handmade', '') RETURNING id;"); - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Stationery & Office Supplies', '', '') RETURNING id;"); - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Tools & Home Improvement', '', '') RETURNING id;"); // Hardware - Heating, cooling, flooring, paint, etc. - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Motor Vehicles, Automotive Parts & Accessories', 'Auto, Tires & Industrial', '') RETURNING id;");//category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Motor Vehicles, Automotive Parts & Accessories', 'Auto, Tires & Industrial', '') RETURNING id;"); - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Beauty & Personal Care', 'Cosmetics;Health, Beauty & Personal Care;Hygiene', '') RETURNING id;"); - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Drugs & Medications', '', '') RETURNING id;"); - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Sports & Outdoors', 'Sporting Equipment;Outdoors & Camping;Hiking;Hunting;Fishing;Biking', '') RETURNING id;"); - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Real Estate, Property & Housing', '', '') RETURNING id;"); - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Luggage & Travel', '', '') RETURNING id;"); - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Business, Industrial & Scientific', '', '') RETURNING id;"); - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Illegal', 'Banned and/or prohibited items', '') RETURNING id;"); - category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Miscellaneous', 'Others;Non-classified', '') RETURNING id;"); - //category_id = database->get_integer("INSERT INTO categories (name, description) VALUES ('Collectables & Art', '') RETURNING id;"); - //category_id = database->get_integer("INSERT INTO categories (name, description) VALUES ('', '') RETURNING id;"); - // NOTE: Categories also act as tags to be used for filtering specific products - } - //------------------------- - if(!database->table_exists("mappings")) { - database->execute("CREATE VIRTUAL TABLE mappings USING fts5(search_term, key, content, tokenize='porter unicode61');"); - } //------------------------- database->execute("COMMIT;"); } @@ -290,7 +209,15 @@ std::string neroshop::Backend::getDatabaseHash() { QVariantList neroshop::Backend::getCategoryList(bool sort_alphabetically) const { QVariantList category_list; - for (const auto& category : predefined_categories) { + std::vector categories = predefined_categories; // Make a copy + + if (sort_alphabetically) { + std::sort(categories.begin(), categories.end(), [](const Category& a, const Category& b) { + return a.name < b.name; + }); + } + + for (const auto& category : categories) { QVariantMap category_object; category_object.insert("id", category.id); category_object.insert("name", QString::fromStdString(category.name)); @@ -298,21 +225,22 @@ QVariantList neroshop::Backend::getCategoryList(bool sort_alphabetically) const category_object.insert("thumbnail", QString::fromStdString(category.thumbnail)); category_list.append(category_object); } - - if(sort_alphabetically == true) { - std::sort(category_list.begin(), category_list.end(), [](const QVariant& a, const QVariant& b) { - return a.toMap()["name"].toString().compare(b.toMap()["name"].toString(), Qt::CaseInsensitive) < 0; - }); - } return category_list; } //---------------------------------------------------------------- -QVariantList neroshop::Backend::getSubCategoryList(int category_id) const { +QVariantList neroshop::Backend::getSubCategoryList(int category_id, bool sort_alphabetically) const { QVariantList subcategory_list; std::vector subcategories = get_subcategories_by_category_id(category_id); + + if (sort_alphabetically) { + std::sort(subcategories.begin(), subcategories.end(), [](const Subcategory& a, const Subcategory& b) { + return a.name < b.name; + }); + } + for (const Subcategory& subcategory : subcategories) { QVariantMap subcategory_obj; subcategory_obj.insert("id", subcategory.id); @@ -373,17 +301,17 @@ bool neroshop::Backend::createFolders() { neroshop::print("\033[1;97;49mcreated path \"" + cache_folder + "\""); } //-------------------------------- - std::string products_folder = cache_folder + "/" + NEROSHOP_CATALOG_FOLDER_NAME; + std::string listings_folder = cache_folder + "/" + NEROSHOP_CATALOG_FOLDER_NAME; // folder with the name should contain all product images for particular listing // datastore/listings/ - if (!neroshop_filesystem::is_directory(products_folder)) { - neroshop::print("Creating directory \"" + products_folder + "\" (^_^) ...", 2); - if (!neroshop_filesystem::make_directory(products_folder)) { - neroshop::print("Failed to create folder \"" + products_folder + "\" (ᵕ人ᵕ)!", 1); + if (!neroshop_filesystem::is_directory(listings_folder)) { + neroshop::print("Creating directory \"" + listings_folder + "\" (^_^) ...", 2); + if (!neroshop_filesystem::make_directory(listings_folder)) { + neroshop::print("Failed to create folder \"" + listings_folder + "\" (ᵕ人ᵕ)!", 1); return false; } - neroshop::print("\033[1;97;49mcreated path \"" + products_folder + "\""); + neroshop::print("\033[1;97;49mcreated path \"" + listings_folder + "\""); } //-------------------------------- // TODO: uncomment this @@ -402,8 +330,68 @@ bool neroshop::Backend::createFolders() { return true; } //---------------------------------------------------------------- -#include -#include +//---------------------------------------------------------------- +bool neroshop::Backend::saveProductThumbnail(const QString& fileName, const QString& listingKey) { + std::string config_path = NEROSHOP_DEFAULT_CONFIGURATION_PATH; + std::string cache_folder = config_path + "/" + NEROSHOP_CACHE_FOLDER_NAME; + std::string listings_folder = cache_folder + "/" + NEROSHOP_CATALOG_FOLDER_NAME; + //---------------------------------------- + // datastore/listings/ + std::string key_folder = listings_folder + "/" + listingKey.toStdString(); + if (!neroshop_filesystem::is_directory(key_folder)) { + neroshop::print("Creating directory \"" + key_folder + "\" (^_^) ...", 2); + if (!neroshop_filesystem::make_directory(key_folder)) { + neroshop::print("Failed to create folder \"" + key_folder + "\" (ᵕ人ᵕ)!", 1); + return false; + } + neroshop::print("\033[1;97;49mcreated path \"" + key_folder + "\""); + } + //---------------------------------------- + // Generate the final destination path + std::string thumbnail_image = "thumbnail.jpg"; + std::string destinationPath = key_folder + "/" + thumbnail_image; + // Check if image already exists in cache so that we do not export the same image more than once + if(!neroshop_filesystem::is_file(destinationPath)) { + // Hopefully the image does not exceed 32 kB in file size :S + QImage sourceImage; + sourceImage.load(fileName); + QSize imageSize = sourceImage.size(); + int maxWidth = 192; // Set the maximum width for the resized image + int maxHeight = 192; // Set the maximum height for the resized image + + // Convert the transparent background to white if necessary + if (sourceImage.hasAlphaChannel()) { + QImage convertedImage = QImage(sourceImage.size(), QImage::Format_RGB32); + convertedImage.fill(Qt::white); + QPainter painter(&convertedImage); + painter.drawImage(0, 0, sourceImage); + painter.end(); + sourceImage = convertedImage; + } + + // Check if the image size is smaller than the maximum size + if (imageSize.width() <= maxWidth && imageSize.height() <= maxHeight) { + // Keep the original image since it's already within the size limits + } else { + // Calculate the new size while maintaining the aspect ratio + QSize newSize = imageSize.scaled(maxWidth, maxHeight, Qt::KeepAspectRatio); + + // Resize the image if it exceeds the maximum dimensions + if (imageSize != newSize) { + sourceImage = sourceImage.scaled(newSize, Qt::KeepAspectRatio, Qt::SmoothTransformation); + } + } + + // Convert the QImage to QPixmap for further processing or saving + QPixmap resizedPixmap = QPixmap::fromImage(sourceImage); + + // Save the resized image + resizedPixmap.save(QString::fromStdString(destinationPath), "JPEG"); + } + + neroshop::print("exported \"" + thumbnail_image + "\" to \"" + cache_folder + "\"", 3); + return true; +} //---------------------------------------------------------------- bool neroshop::Backend::saveProductImage(const QString& fileName, const QString& listingKey) { std::string config_path = NEROSHOP_DEFAULT_CONFIGURATION_PATH; @@ -414,18 +402,18 @@ bool neroshop::Backend::saveProductImage(const QString& fileName, const QString& std::string image_name = image_file.substr(image_file.find_last_of("\\/") + 1);// get filename from path (complete base name) //---------------------------------------- // datastore/listings/ - std::string products_folder = listings_folder + "/" + listingKey.toStdString(); - if (!neroshop_filesystem::is_directory(products_folder)) { - neroshop::print("Creating directory \"" + products_folder + "\" (^_^) ...", 2); - if (!neroshop_filesystem::make_directory(products_folder)) { - neroshop::print("Failed to create folder \"" + products_folder + "\" (ᵕ人ᵕ)!", 1); + std::string key_folder = listings_folder + "/" + listingKey.toStdString(); + if (!neroshop_filesystem::is_directory(key_folder)) { + neroshop::print("Creating directory \"" + key_folder + "\" (^_^) ...", 2); + if (!neroshop_filesystem::make_directory(key_folder)) { + neroshop::print("Failed to create folder \"" + key_folder + "\" (ᵕ人ᵕ)!", 1); return false; } - neroshop::print("\033[1;97;49mcreated path \"" + products_folder + "\""); + neroshop::print("\033[1;97;49mcreated path \"" + key_folder + "\""); } //---------------------------------------- // Generate the final destination path - std::string destinationPath = products_folder + "/" + image_name; + std::string destinationPath = key_folder + "/" + image_name; // Check if image already exists in cache so that we do not export the same image more than once if(!neroshop_filesystem::is_file(destinationPath)) { // Image Loader crashes when image resolution is too large (ex. 4096 pixels wide) so we need to scale it!! @@ -470,7 +458,6 @@ QVariantMap neroshop::Backend::uploadProductImage(const QString& fileName, int i product_image_file.seekg(0, std::ios::end); size_t size = static_cast(product_image_file.tellg()); // in bytes // Limit product image size to 12582912 bytes (12 megabyte) - // Todo: Database cannot scale to billions of users if I am storing blobs so I'll have to switch to text later const int max_bytes = 12582912; double kilobytes = max_bytes / 1024.0; double megabytes = kilobytes / 1024.0; @@ -1017,7 +1004,6 @@ QVariantList neroshop::Backend::getListings(ListingSorting sorting) { listing.insert("product_description", QString::fromStdString(product_obj["description"].get())); listing.insert("product_category_id", get_category_id_by_name(product_obj["category"].get())); //listing.insert("", QString::fromStdString(product_obj[""].get())); - //listing.insert("", QString::fromStdString(product_obj[""].get())); if (product_obj.contains("images") && product_obj["images"].is_array()) { const auto& images_array = product_obj["images"]; for (const auto& image : images_array) { @@ -1033,6 +1019,9 @@ QVariantList neroshop::Backend::getListings(ListingSorting sorting) { } listing.insert("product_images", product_images); } + if (product_obj.contains("thumbnail") && product_obj["thumbnail"].is_string()) { + listing.insert("product_thumbnail", QString::fromStdString(product_obj["thumbnail"].get())); + } } catalog.append(listing); } diff --git a/src/gui/backend.hpp b/src/gui/backend.hpp index 62a9a93..6754ad4 100644 --- a/src/gui/backend.hpp +++ b/src/gui/backend.hpp @@ -37,6 +37,9 @@ public: Q_INVOKABLE QString urlToLocalFile(const QUrl& url) const; Q_INVOKABLE void copyTextToClipboard(const QString& text); + + QString imageToBase64(const QImage& image); // un-tested + QImage base64ToImage(const QString& base64Data); // un-tested Q_INVOKABLE QStringList getCurrencyList() const; Q_INVOKABLE int getCurrencyDecimals(const QString& currency) const; @@ -48,7 +51,7 @@ public: // TODO: Use Q_ENUM for sorting in order by a specific column (e.e Sort.Name, Sort.Id) Q_INVOKABLE QVariantList getCategoryList(bool sort_alphabetically = false) const; - Q_INVOKABLE QVariantList getSubCategoryList(int category_id) const; + Q_INVOKABLE QVariantList getSubCategoryList(int category_id, bool sort_alphabetically = false) const; Q_INVOKABLE int getCategoryIdByName(const QString& category_name) const; Q_INVOKABLE int getSubCategoryIdByName(const QString& subcategory_name) const; Q_INVOKABLE int getCategoryProductCount(int category_id) const; // returns number of products that fall under a specific category @@ -75,9 +78,9 @@ public: // Products should be registered so that sellers can list pre-existing products without the need to duplicate a product which is unnecessary and can make the database bloated Q_INVOKABLE bool createFolders(); - Q_INVOKABLE QVariantMap uploadProductImage(const QString& filename, int image_id); + Q_INVOKABLE QVariantMap uploadProductImage(const QString& filename, int image_id); // constructs image object rather than upload it Q_INVOKABLE bool saveProductImage(const QString& fileName, const QString& listingKey); - + Q_INVOKABLE bool saveProductThumbnail(const QString& fileName, const QString& listingKey); Q_INVOKABLE int getProductStarCount(const QString& product_id); // getProductRatingsCount Q_INVOKABLE int getProductStarCount(const QString& product_id, int star_number); diff --git a/src/gui/user_controller.cpp b/src/gui/user_controller.cpp index 51b874c..44df5ea 100644 --- a/src/gui/user_controller.cpp +++ b/src/gui/user_controller.cpp @@ -82,6 +82,7 @@ int quantity, double price, const QString& currency, const QString& condition, c if(imageMap.contains("name")) image.name = imageMap.value("name").toString().toStdString(); if(imageMap.contains("size")) image.size = imageMap.value("size").toInt(); if(imageMap.contains("id")) image.id = imageMap.value("id").toInt(); + if(imageMap.contains("source")) image.source = imageMap.value("source").toString().toStdString(); //if(imageMap.contains("")) imagesVector.push_back(image); @@ -169,6 +170,22 @@ void neroshop::UserController::rateSeller(const QString& seller_id, int score, c } //---------------------------------------------------------------- //---------------------------------------------------------------- +void neroshop::UserController::addToFavorites(const QString& listing_key) { + if (!_user) throw std::runtime_error("neroshop::User is not initialized"); + _user->add_to_favorites(listing_key.toStdString()); +} +//---------------------------------------------------------------- +void neroshop::UserController::removeFromFavorites(const QString& listing_key) { + if (!_user) throw std::runtime_error("neroshop::User is not initialized"); + _user->remove_from_favorites(listing_key.toStdString()); +} +//---------------------------------------------------------------- +bool neroshop::UserController::hasFavorited(const QString& listing_key) { + if (!_user) throw std::runtime_error("neroshop::User is not initialized"); + return _user->has_favorited(listing_key.toStdString()); +} +//---------------------------------------------------------------- +//---------------------------------------------------------------- void neroshop::UserController::uploadAvatar(const QString& filename) { if (!_user) throw std::runtime_error("neroshop::User is not initialized"); diff --git a/src/gui/user_controller.hpp b/src/gui/user_controller.hpp index 64bdc30..b59b3a5 100644 --- a/src/gui/user_controller.hpp +++ b/src/gui/user_controller.hpp @@ -69,8 +69,10 @@ public: Q_INVOKABLE void createOrder(const QString& shipping_address); Q_INVOKABLE void rateItem(const QString& product_id, int stars, const QString& comments);//, const QString& signature); Q_INVOKABLE void rateSeller(const QString& seller_id, int score, const QString& comments);//, const QString& signature); - //Q_INVOKABLE void addToFavorites(); - //Q_INVOKABLE void removeFromFavorites(); + + Q_INVOKABLE void addToFavorites(const QString& listing_key); + Q_INVOKABLE void removeFromFavorites(const QString& listing_key); + Q_INVOKABLE bool hasFavorited(const QString& listing_key); //Q_INVOKABLE void setID(const QString& id); //Q_INVOKABLE void setWallet(neroshop::WalletController * wallet); // get the actual wallet from the controller then set it as the wallet diff --git a/tests/request/request b/tests/request/request index 9b1cf67c92595cbc7c7268c8f0ede41a2c268222..9153f9519c6ddb8cc5b62057fb527d58b1a92cbe 100644 GIT binary patch delta 763 zcmYk4T}TvB6vyuwcbhi+tku<2b`5K?2iFCmD5pn642@*y zNtiq?l{w2LS|~^*tp~GVjy$QXczxkL@4D3G_R7If=W(^P?cT4>BVs7eQ%i^WZEZgV z#kAH!%1@sN)LnO1zu(MQ(Q_BwP*Ud7wRuXWt(5t}qO>(!@Peo7^}>s}3TwFZtk@Eq zB8rOnhOb3c=Q?_Ny3U{O-M4L1;LqxJcKwVAcXKaYnffH<8lKtrMao)YYyfNmy|Yrz zA-@iOL>!w#9oRmPIFGSWa1Bg@apYM>%0BQGI0gI^2|iNT zUbeg>U)l}e$ zI3i{i77r?V!3O0J>arExNJn|X(C8$;WB4IC4nM|c9KUL4&H0!|y*;15-Fe%|ZpVGh zh0ooJ`9!hLohwF8{TTWz-eGDq%af*`ruiuR0-rRs8Wm$J`@b*hv^ly{&|T*x5kD>R z5c~xmgMT0rkqI~d;R}d{))6>!rkQPq}Vi T!&o^VN7Nn3M%`jH@lg2J<{4n|wLLY={!9E2`h+ZfJgQ7$&p;;>tf-sDy|DCY9aL(_Z^PKa4&OO}N3dL4v z%1d^)jin#A(Mle2gj+Vl40%*qEu5X&>&>5eHX3P|*x6K7(0G1*=72F~vAZe4?;3H5ZiM?p?Fxn{dPGU{gGR(~1Ql(rNPuPa$j&Ci9!>cJlE_)kQ)6 zT&;BeDPgRnWUDRv24*)!Op9~7rj)HOVE{x-&&|IosX1Gu!>HS=2qRs^1DZl!KBT!I zc@O?JkEi{6O=(VHFV)b@bX#_l*%(@;RiyrYP z{0ly*D`j7>Kj;5_ZRxhO?i_UI`5J0pa)-}F?YtI#*a-NZSdE$^F)NMm*}w@38?_y8 qtQ0fo*r8(T;mv&>`@g?;sBgc$UNx^^te(dajrlgCF=M^&f%F$GHu~ZK diff --git a/tests/request/request.cpp b/tests/request/request.cpp index b1921a5..fb0f03b 100644 --- a/tests/request/request.cpp +++ b/tests/request/request.cpp @@ -23,7 +23,7 @@ int main() { "jsonrpc": "2.0", "method": "query", "params": { - "sql": "SELECT * FROM categories WHERE name = 'Food & Beverages';" + "sql": "SELECT * FROM mappings;" } })";//"{\"id\": \"5135958352\", \"jsonrpc\": \"2.0\", \"method\": \"query\", \"params\": {\"sql\": \"SELECT * FROM users WHERE name = 'layter'\"}}"; diff --git a/tests/request/request.sh b/tests/request/request.sh index bc29b16..c4252c6 100644 --- a/tests/request/request.sh +++ b/tests/request/request.sh @@ -1,6 +1,6 @@ #!/bin/bash curl -X POST \ -H 'Content-Type: application/json' \ - -d '{"id": "5135958352", "jsonrpc": "2.0", "method": "query", "params": {"sql": "SELECT * FROM categories WHERE name = '\''Food & Beverages'\''"}}' \ + -d '{"id": "5135958352", "jsonrpc": "2.0", "method": "query", "params": {"sql": "SELECT * FROM mappings;"}}' \ http://127.0.0.1:50882 # -d @./request.json \