add favorites (wishlist) feature

pull/165/head
larteyoh 10 months ago
parent 7df4503d48
commit 45a3ade1cb

@ -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 {

@ -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)

@ -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() })

@ -44,26 +44,40 @@ const std::vector<Category> 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<Subcategory> 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<unsigned int>(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<unsigned int>(predefined_categories.size() + 0), predefined_categories[5].name, predefined_categories[5].description, predefined_categories[5].thumbnail, 7 },
{ static_cast<unsigned int>(predefined_categories.size() + 1), predefined_categories[5].name, predefined_categories[5].description, predefined_categories[5].thumbnail, 8 },
{ static_cast<unsigned int>(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<unsigned int>(predefined_categories.size() + 3), predefined_categories[25].name, predefined_categories[25].description, predefined_categories[25].thumbnail, 0 },
{ static_cast<unsigned int>(predefined_categories.size() + 4), predefined_categories[25].name, predefined_categories[25].description, predefined_categories[25].thumbnail, 5 },
{ static_cast<unsigned int>(predefined_categories.size() + 5), predefined_categories[25].name, predefined_categories[25].description, predefined_categories[25].thumbnail, 6 },
{ static_cast<unsigned int>(predefined_categories.size() + 6), predefined_categories[25].name, predefined_categories[25].description, predefined_categories[25].thumbnail, 7 },
{ static_cast<unsigned int>(predefined_categories.size() + 7), predefined_categories[25].name, predefined_categories[25].description, predefined_categories[25].thumbnail, 8 },
{ static_cast<unsigned int>(predefined_categories.size() + 8), predefined_categories[25].name, predefined_categories[25].description, predefined_categories[25].thumbnail, 19 },
{ static_cast<unsigned int>(predefined_categories.size() + 9), predefined_categories[25].name, predefined_categories[25].description, predefined_categories[25].thumbnail, 23 },
{ static_cast<unsigned int>(predefined_categories.size() + 10), predefined_categories[25].name, predefined_categories[25].description, predefined_categories[25].thumbnail, 24 },
//{ 0, , <category_id_goes_here> },
// All categories
{ static_cast<unsigned int>(predefined_categories.size() + 11), predefined_categories[0].name, predefined_categories[0].description, predefined_categories[0].thumbnail, 25 },
{ static_cast<unsigned int>(predefined_categories.size() + 12), predefined_categories[5].name, predefined_categories[5].description, predefined_categories[5].thumbnail, 25 },
{ static_cast<unsigned int>(predefined_categories.size() + 13), predefined_categories[6].name, predefined_categories[6].description, predefined_categories[6].thumbnail, 25 },
{ static_cast<unsigned int>(predefined_categories.size() + 14), predefined_categories[7].name, predefined_categories[7].description, predefined_categories[7].thumbnail, 25 },
{ static_cast<unsigned int>(predefined_categories.size() + 15), predefined_categories[8].name, predefined_categories[8].description, predefined_categories[8].thumbnail, 25 },
{ static_cast<unsigned int>(predefined_categories.size() + 16), predefined_categories[19].name, predefined_categories[19].description, predefined_categories[19].thumbnail, 25 },
{ static_cast<unsigned int>(predefined_categories.size() + 17), predefined_categories[23].name, predefined_categories[23].description, predefined_categories[23].thumbnail, 25 },
{ static_cast<unsigned int>(predefined_categories.size() + 18), predefined_categories[24].name, predefined_categories[24].description, predefined_categories[24].thumbnail, 25 },
{ static_cast<unsigned int>(predefined_categories.size() + 19), predefined_categories[7].name, predefined_categories[7].description, predefined_categories[7].thumbnail, 5 },
{ static_cast<unsigned int>(predefined_categories.size() + 20), predefined_categories[8].name, predefined_categories[8].description, predefined_categories[8].thumbnail, 5 },
{ static_cast<unsigned int>(predefined_categories.size() + 21), predefined_categories[9].name, predefined_categories[9].description, predefined_categories[9].thumbnail, 5 },
};
//-----------------------------------------------------------------------------

@ -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
};
}

@ -137,7 +137,7 @@ std::pair<std::string, std::string/*std::vector<uint8_t>*/> 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";
}

@ -668,10 +668,10 @@ neroshop::User * neroshop::Seller::on_login(const neroshop::Wallet& wallet) { //
dynamic_cast<Seller *>(user)->set_account_type(UserAccountType::Seller);
//-------------------------------
/*// load orders
dynamic_cast<Seller *>(user)->load_orders();
dynamic_cast<Seller *>(user)->load_orders();*/
// load wishlists
dynamic_cast<Seller *>(user)->load_favorites();
// load customer_orders
/*// load customer_orders
static_cast<Seller *>(user)->load_customer_orders();*/
// Load cart (into memory)
user->get_cart()->load(user->get_id());

@ -3,6 +3,7 @@
#include <openssl/evp.h>
#include <openssl/bio.h>
#include <openssl/buffer.h>
#include <fstream>
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<char> 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<int>(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<char> 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<int>(encoded.length()));
BIO_push(b64, mem);
// Determine the size of the decoded data
size_t max_decoded_length = (encoded.length() * 3) / 4;
std::vector<char> decoded_data(max_decoded_length);
// Decode the data
int decoded_length = BIO_read(b64, decoded_data.data(), static_cast<int>(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!";

@ -2,10 +2,14 @@
#include <iostream>
#include <string>
#include <vector>
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<char> base64_image_decode(const std::string& encoded);
}

@ -11,7 +11,7 @@
#include <fstream>
////////////////////
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<Cart>(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<neroshop::Product>(product_id));//(std::unique_ptr<neroshop::Product>(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<const char *> 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<neroshop::Product>(product_id));//(std::unique_ptr<neroshop::Product>(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<const char *>(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::Order *> 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::Product *> neroshop::User::get_favorites_list() const {
std::vector<neroshop::Product *> 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<std::string> 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)
}
////////////////////
////////////////////

@ -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<neroshop::Order *> 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<neroshop::Product *> get_favorites_list() const;
std::vector<std::string> 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> cart;
std::vector<std::shared_ptr<neroshop::Order>> order_list;
std::vector<std::shared_ptr<neroshop::Product>> 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<std::string> favorites;
std::string get_private_key() const;
};
}

@ -14,6 +14,13 @@
#include <QProcess> // Note: QProcess is not supported on VxWorks, iOS, tvOS, or watchOS.
#include <QUuid>
#include <QDateTime>
#include <QImage>
#include <QPixmap>
#include <QByteArray>
#include <QBuffer>
#include <QPainter>
#include <QImageReader>
#include <QImageWriter>
#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 <future>
#include <thread>
@ -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<Category> 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<Subcategory> 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 <listing_id> 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 <QImage>
#include <QPixmap>
//----------------------------------------------------------------
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/<listing_key>
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/<listing_key>
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<int>(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<std::string>()));
listing.insert("product_category_id", get_category_id_by_name(product_obj["category"].get<std::string>()));
//listing.insert("", QString::fromStdString(product_obj[""].get<std::string>()));
//listing.insert("", QString::fromStdString(product_obj[""].get<std::string>()));
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<std::string>()));
}
}
catalog.append(listing);
}

@ -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);

@ -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");

@ -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

Binary file not shown.

@ -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'\"}}";

@ -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 \

Loading…
Cancel
Save