diff --git a/CMakeLists.txt b/CMakeLists.txt index be87240..58362ab 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -4,11 +4,15 @@ project(checks-parser VERSION 0.1 LANGUAGES CXX) option(BUILD_TRANSLATIONS "Build translations?" ON) +include(FetchContent) + +SET(CMAKE_BUILD_TYPE Debug) + set(CMAKE_AUTOUIC ON) set(CMAKE_AUTOMOC ON) set(CMAKE_AUTORCC OFF) set(CMAKE_INCLUDE_CURRENT_DIR ON) -set(CMAKE_AUTOUIC_SEARCH_PATHS Designer) +set(CMAKE_AUTOUIC_SEARCH_PATHS scenes) set(CMAKE_CXX_STANDARD 17) set(CMAKE_CXX_STANDARD_REQUIRED ON) @@ -17,39 +21,43 @@ find_package(Qt5Core REQUIRED) find_package(Qt5 REQUIRED COMPONENTS LinguistTools) find_package(Qt5Gui REQUIRED) find_package(Qt5Widgets REQUIRED) +find_package(Qt5UiTools REQUIRED) set(PROJECT_SOURCES main.cpp - mainwindow.cpp - mainwindow.h - mainwindow.ui + mainwindow.h mainwindow.cpp scenes/mainwindow.ui goods/goods.h goods/goods.cpp check/check.h check/check.cpp parser/parser.h parser/parser.cpp parser/module.h parser/module.cpp - outputdialog.h outputdialog.cpp outputdialog.ui output/output_options.h output/output_options.cpp utils/utils.h utils/utils.cpp image/checkimage.h image/checkimage.cpp net/net.h net/net.cpp settings/settings.h settings/settings.cpp - settingsdialog.h settingsdialog.cpp settingsdialog.ui - adjustpicturedialog.h adjustpicturedialog.cpp adjustpicturedialog.ui - image_redactor/imageredactor.h image_redactor/imageredactor.cpp - solvecaptchadialog.h solvecaptchadialog.cpp solvecaptchadialog.ui exceptions/ofdrequestexception.h exceptions/ofdrequestexception.cpp + + emailtextscene.h emailtextscene.cpp scenes/emailtextscene.ui + ocrscene.h ocrscene.cpp scenes/ocrscene.ui + ofdscene.h ofdscene.cpp scenes/ofdscene.ui + outputdialog.h outputdialog.cpp scenes/outputdialog.ui + adjustpicturedialog.h adjustpicturedialog.cpp scenes/adjustpicturedialog.ui + image_redactor/imageredactor.h image_redactor/imageredactor.cpp + solvecaptchadialog.h solvecaptchadialog.cpp scenes/solvecaptchadialog.ui ) set(TRANSLATION_SOURCES main.cpp - mainwindow.cpp mainwindow.h mainwindow.ui - outputdialog.cpp outputdialog.h outputdialog.ui - settingsdialog.cpp settingsdialog.h settingsdialog.ui - solvecaptchadialog.cpp solvecaptchadialog.h solvecaptchadialog.ui - adjustpicturedialog.cpp adjustpicturedialog.h adjustpicturedialog.ui + mainwindow.cpp mainwindow.h scenes/mainwindow.ui + emailtextscene.cpp emailtextscene.h scenes/emailtextscene.ui + ocrscene.cpp ocrscene.h scenes/ocrscene.ui + ofdscene.cpp ofdscene.h scenes/ofdscene.ui + outputdialog.h outputdialog.cpp scenes/outputdialog.ui + adjustpicturedialog.h adjustpicturedialog.cpp scenes/adjustpicturedialog.ui + solvecaptchadialog.h solvecaptchadialog.cpp scenes/solvecaptchadialog.ui ) set(TS_FILES @@ -66,7 +74,23 @@ if (BUILD_TRANSLATIONS) add_dependencies(resources translations) endif() +# Media QRC +configure_file(${CMAKE_CURRENT_SOURCE_DIR}/media.qrc ${CMAKE_CURRENT_BINARY_DIR}/media.qrc COPYONLY) +file(COPY ${CMAKE_CURRENT_SOURCE_DIR}/assets DESTINATION ${CMAKE_CURRENT_BINARY_DIR}) +qt5_add_resources(MEDIAQRC ${CMAKE_CURRENT_BINARY_DIR}/media.qrc) +add_custom_target(mediaresource ALL DEPENDS ${MEDIAQRC}) + +#Scenes QRC +configure_file(${CMAKE_CURRENT_SOURCE_DIR}/scenes.qrc ${CMAKE_CURRENT_BINARY_DIR}/scenes.qrc COPYONLY) +file(COPY ${CMAKE_CURRENT_SOURCE_DIR}/scenes DESTINATION ${CMAKE_CURRENT_BINARY_DIR}) +qt5_add_resources(SCENESQRC ${CMAKE_CURRENT_BINARY_DIR}/scenes.qrc) +add_custom_target(scenessource ALL DEPENDS ${SCENESQRC}) + set(SOURCES "") + +list(APPEND SOURCES ${MEDIAQRC}) +list(APPEND SOURCES ${SCENESQRC}) + if (BUILD_TRANSLATIONS) list(APPEND SOURCES ${TRANSLATIONQRC}) endif() @@ -83,7 +107,7 @@ else() ) endif() -target_link_libraries(checks-parser PRIVATE Qt5::Widgets) +target_link_libraries(checks-parser PRIVATE Qt5::Widgets Qt5::UiTools) target_include_directories(checks-parser PUBLIC ${CMAKE_CURRENT_SOURCE_DIR}/image_redactor) @@ -104,20 +128,28 @@ install(TARGETS checks-parser RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR} ) - if(WIN32) #???? set(OpenCV_DIR /usr/local/lib/cmake/opencv4) endif() +FetchContent_Declare(httplib SYSTEM + GIT_REPOSITORY https://github.com/yhirose/cpp-httplib + GIT_TAG 2eaa2ea64f9fb12773306534d461d9ed63cb76b6 # v0.14.1 + GIT_SHALLOW TRUE) +FetchContent_MakeAvailable(httplib) + find_package(OpenCV REQUIRED COMPONENTS core imgproc imgcodecs) include_directories( ${OpenCV_INCLUDE_DIRS} ) +target_include_directories(checks-parser PUBLIC ${OpenCV_INCLUDE_DIRS}) target_link_libraries(checks-parser PRIVATE -lzbar) target_link_libraries(checks-parser PRIVATE -ltesseract) target_link_libraries(checks-parser PRIVATE -lcurl) -target_link_libraries(checks-parser PRIVATE ${OpenCV_LIBS} ) +target_link_libraries(checks-parser PRIVATE ${OpenCV_LIBS}) +target_link_libraries(checks-parser PRIVATE httplib) + if (CMAKE_CXX_COMPILER_VERSION VERSION_LESS 8) target_link_libraries(checks-parser PRIVATE -lstdc++fs) endif() diff --git a/README.ru.md b/README.ru.md index ca2bfe3..85adc56 100644 --- a/README.ru.md +++ b/README.ru.md @@ -4,25 +4,25 @@ Чек парсер - это приложение, позволяющее доставать содержимое чека и переводить его в .csv файлы. !!!НА ДАННЫЙ МОМЕНТ ПОДДЕРЖИВАЮТСЯ ТОЛЬКО РУССКИЕ ЧЕКИ!!! -Чтобы узнать, почему [смотрите здесь](https://git.foxarmy.org/leca/checks-parser/src/branch/master/README.ru.md#checks-from-different-countries) +Чтобы узнать почему [смотрите здесь](https://git.foxarmy.org/leca/checks-parser/src/branch/master/README.ru.md#checks-from-different-countries) # Пользование Для более детального описания, пожалуйста, обратитесь к [вики](https://git.foxarmy.org/leca/checks-parser/wiki/Description-%5BRU%5D) ### Ввод данных Доступны следующие способы ввода данных: -* Через изображениие (используется OCR(Optical Character Recognition, Оптическое распознавание символов) чтобы прочитать содержимое чека. Изображение чека должно быть контрастным и выровненным (текст обязан быть перпендикулярным к границам изображения) хорошо, чтобы нормально прочитаться.) OCR - не волшебная палочка :( -* Через просто текст, скопированный из эл. письма. Просто скопируйте и вставьте текст с вашего письма, выберите подходящий магазин (автодетект в планах!) и парсите. -* Через QRCode на чеке (этот метод запрашивает данные у ОФД (Оператор Фискальных Данных) (В данном конкретном случае, к ofd.ru)). +* Через изображениие (используется OCR(Optical Character Recognition, Оптическое распознавание символов), чтобы прочитать содержимое чека. Изображение чека должно быть контрастным и выровненным (текст обязан быть перпендикулярным к границам изображения) хорошо, чтобы нормально прочитаться. OCR - не волшебная палочка :( +* Через простой текст, скопированный из эл. письма. Скопируйте и вставьте текст с вашего письма, выберите подходящий магазин (автодетект в планах!) и парсите. +* Через QRCode на чеке (этот метод запрашивает данные у ОФД (Оператор Фискальных Данных), в данном конкретном случае, к ofd.ru). ### Вывод данных -На начальном этапе разработки программы, Я задумывался о 3 или более форматов вывода: csv, xlsx и ods. Но по мере разработки, я понял, что большинство современных табличных процессоров (электронных таблиц) способны импортировать в себя csv гораздо лучше, чем смог бы написать я сам. Так что я решил не делать вывод во все остальные форматы, кроме csv. +На начальном этапе разработки программы, Я задумывался о 3-х или более форматов вывода: csv, xlsx и ods. Но, по мере разработки, я понял, что большинство современных табличных процессоров (электронных таблиц) способны импортировать в себя csv гораздо лучше, чем смог бы написать я сам. Так что я решил не делать вывод во все остальные форматы, кроме csv. -Чтобы экспортировать, вам нужно указать путь до файла, если вы желаете, вы можете изменить порядок и/или переименовать (алиасы) столбцы, выбрать печатать или не печатать заголовок (алиасы столбцов) и "итого" +Чтобы экспортировать вам нужно: указать путь до файла, если вы желаете, вы можете изменить порядок и/или переименовать (алиасы) столбцы, выбрать печатать или не печатать заголовок (алиасы столбцов) и "итого". # Установка ## Сборка из исходников -В целом, вам нужно установить следующие зависимости чтобы собрать приложение (я предполагаю, что вы уже имеете на системе базовые пакеты вроде cmake, make, gcc, git и так далее): +В целом, вам нужно установить следующие зависимости, чтобы собрать приложение (я предполагаю, что вы уже имеете на системе базовые пакеты вроде cmake, make, gcc, git и так далее): * tesseract (также вам нужно будет установить языковой пакет для него, например tesseract-data-rus на Arch Linux или tesseract-ocr-rus на Debian Linux.) * opencv * zbar diff --git a/assets/icons/OCR.svg b/assets/icons/OCR.svg new file mode 100644 index 0000000..2157b4d --- /dev/null +++ b/assets/icons/OCR.svg @@ -0,0 +1,131 @@ + + + + + + + + + + + + + + + + + + + + + + + + Receipt + + diff --git a/assets/icons/OFD.svg b/assets/icons/OFD.svg new file mode 100644 index 0000000..1d3f058 --- /dev/null +++ b/assets/icons/OFD.svg @@ -0,0 +1,181 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + ofd.ru + ФНС + + + + + Checks parser + + diff --git a/assets/icons/email-text.svg b/assets/icons/email-text.svg new file mode 100644 index 0000000..8018c7c --- /dev/null +++ b/assets/icons/email-text.svg @@ -0,0 +1,285 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Receipt + + + diff --git a/assets/icons/using_binary_eye.svg b/assets/icons/using_binary_eye.svg new file mode 100644 index 0000000..cb932be --- /dev/null +++ b/assets/icons/using_binary_eye.svg @@ -0,0 +1,219 @@ + + + +Checks parser diff --git a/check/check.cpp b/check/check.cpp index 6a1881a..587d0eb 100644 --- a/check/check.cpp +++ b/check/check.cpp @@ -5,10 +5,16 @@ Check::Check() {} void Check::add_goods(Goods goods) { this->goods.push_back(goods); } +void Check::add_goods(std::vector &goods) { + for (auto g : goods) { + this->goods.push_back(g); + } +} + double Check::calculae_total_price() { double total = 0.0; - for (Goods g : this->goods) { + for (Goods &g : goods) { total += g.calculate_total_price(); } diff --git a/check/check.h b/check/check.h index a7b8299..01af62b 100644 --- a/check/check.h +++ b/check/check.h @@ -9,6 +9,7 @@ class Check { public: Check(); void add_goods(Goods); + void add_goods(std::vector &goods); double calculae_total_price(); diff --git a/emailtextscene.cpp b/emailtextscene.cpp new file mode 100644 index 0000000..5c70c11 --- /dev/null +++ b/emailtextscene.cpp @@ -0,0 +1,45 @@ +#include "emailtextscene.h" +#include "ui_emailtextscene.h" +#include +#include +#include +#include + +EmailTextScene::EmailTextScene(QWidget *parent) + : QWidget(parent) + , ui(new Ui::EmailTextScene) { + ui->setupUi(this); + + auto modules = parser.get_modules_names(); + + for (auto &module : modules) { + ui->store_combo_box->addItem(QString::fromStdString(module)); + } +} + +EmailTextScene::~EmailTextScene() { + delete ui; +} + +void EmailTextScene::on_parse_button_clicked() { + std::wstring checkContent = ui->check_content->toPlainText().toStdWString(); + parser.set_module(parser.search_modules()[ui->store_combo_box->currentIndex()]); + + std::vector goods = parser.parse(checkContent); + if (goods.size() == 0) { + QMessageBox infoDialog; + infoDialog.setText(tr("An error has occured. Check was matched incorrectly. Vector sizes are different. Please, contact the developer.")); + infoDialog.setIcon(QMessageBox::Critical); + infoDialog.setWindowTitle(tr("Error in parsing")); + infoDialog.exec(); + return; + } + + Check check; + check.add_goods(goods); + + OutputDialog d(this, check); + d.show(); + d.exec(); +} + diff --git a/emailtextscene.h b/emailtextscene.h new file mode 100644 index 0000000..4a8739a --- /dev/null +++ b/emailtextscene.h @@ -0,0 +1,27 @@ +#ifndef EMAILTEXTSCENE_H +#define EMAILTEXTSCENE_H + +#include "parser/parser.h" +#include + +namespace Ui { +class EmailTextScene; +} + +class EmailTextScene : public QWidget +{ + Q_OBJECT + +public: + explicit EmailTextScene(QWidget *parent = nullptr); + ~EmailTextScene(); + +private slots: + void on_parse_button_clicked(); + +private: + Ui::EmailTextScene *ui; + Parser parser; +}; + +#endif // EMAILTEXTSCENE_H diff --git a/image/checkimage.cpp b/image/checkimage.cpp index caddf3d..4210ca6 100644 --- a/image/checkimage.cpp +++ b/image/checkimage.cpp @@ -1,7 +1,6 @@ +#include #include #include -#include -#include #include "checkimage.h" CheckImage::CheckImage(std::string path) { diff --git a/main.cpp b/main.cpp index de82bd9..69ab8b6 100644 --- a/main.cpp +++ b/main.cpp @@ -12,20 +12,162 @@ # include using namespace std::filesystem; #endif +#include #include +#include #include #include +#include +#include +#include +#include +#include + +static QWidget *loadUI(QWidget *parent, std::string filename) { + QUiLoader loader; + + QFile file(QString::fromStdString(filename)); + file.open(QIODevice::ReadOnly); + + return loader.load(&file, parent); +} int main(int argc, char *argv[]) { - curl_global_init(CURL_GLOBAL_ALL); - std::string program_data_path = get_path_relative_to_home(".local/share/checks_parser"); - create_directories(program_data_path); + QApplication app(argc, argv); std::string settings_file_path = get_path_relative_to_home(".local/share/checks_parser/settings.json"); Settings s(settings_file_path); + + QTranslator translator; + QString lang = "en_US"; + + if (s.get_all_settings().contains("language")) { + lang = QString::fromStdString(s.get_all_settings()["language"]); + } else if (translator.load(":/translation/"+QLocale::system().name()+".qm")) { + lang = QLocale::system().name(); + } else { + lang = QString::fromStdString("en_US"); + } + + std::cout << "Using locale: " << lang.toStdString() << std::endl; + + translator.load(":/translation/" + lang + ".qm"); + app.installTranslator(&translator); + + QUiLoader loader; + + QWidget *window = new QWidget(); + QStackedLayout *sceneLayout = new QStackedLayout; + + // Main Window setup + QWidget *mainwindowscene = loadUI(window, ":/scenes/scenes/mainwindow.ui"); + + // Main Window buttons setup + QPushButton *text_from_email_button = ((MainWindow *)mainwindowscene)->findChild("text_from_email_button"); + QPushButton *ocr_button = ((MainWindow *)mainwindowscene)->findChild("ocr_button"); + QPushButton *ofd_button = ((MainWindow *)mainwindowscene)->findChild("ofd_button"); + + QObject::connect(text_from_email_button, &QPushButton::clicked, [&]() { + // Text from email scene + sceneLayout->setCurrentIndex(1); + sceneLayout->widget(1)->show(); + }); + + QObject::connect(ocr_button, &QPushButton::clicked, [&]() { + // OCR scene + sceneLayout->setCurrentIndex(2); + sceneLayout->widget(2)->show(); + }); + + QObject::connect(ofd_button, &QPushButton::clicked, [&]() { + // OFD scene + sceneLayout->setCurrentIndex(3); + sceneLayout->widget(3)->show(); + }); + + EmailTextScene *emailTextScene = new EmailTextScene(); + OCRScene *ocrscene = new OCRScene(); + OFDScene *ofdscene = new OFDScene(); + + // ofdscene->startHttpServer(); + // get_local_ip_address(); + + + sceneLayout->addWidget(mainwindowscene); + sceneLayout->addWidget(emailTextScene); + sceneLayout->addWidget(ocrscene); + sceneLayout->addWidget(ofdscene); + + //Setting all back buttons + for (uint32_t sceneIndex = 0; sceneIndex < sceneLayout->count(); sceneIndex ++) { + auto scene = sceneLayout->widget(sceneIndex); + + QPushButton *back_button = scene->findChild("back_button"); + if (back_button == nullptr) continue; + + QObject::connect(back_button, &QPushButton::clicked, [&]() { + sceneLayout->setCurrentIndex(0); + }); + } + + + + window->setLayout(sceneLayout); + window->show(); + + app.exec(); + + return 0; + + // QApplication app(argc, argv); + + // QWidget *window = new QWidget; + // QStackedLayout *stackedLayout = new QStackedLayout; + + // QWidget *scene1 = new QWidget; + // QWidget *scene2 = new QWidget; + + // // Add some widgets to each scene + // QPushButton *button1 = new QPushButton("Switch to Scene 2"); + // scene1->setLayout(new QVBoxLayout); + // scene1->layout()->addWidget(button1); + + // QPushButton *button2 = new QPushButton("Switch to Scene 1"); + // scene2->setLayout(new QVBoxLayout); + // scene2->layout()->addWidget(button2); + + // // Add the scenes to the stacked layout + // stackedLayout->addWidget(scene1); + // stackedLayout->addWidget(scene2); + + // // Set the layout of the window + // window->setLayout(stackedLayout); + + // // Connect the buttons to switch scenes + // QObject::connect(button1, &QPushButton::clicked, [&]() { + // stackedLayout->setCurrentIndex(1); + // }); + + // QObject::connect(button2, &QPushButton::clicked, [&]() { + // stackedLayout->setCurrentIndex(0); + // }); + + // window->show(); + // app.exec(); + + // return 0; + curl_global_init(CURL_GLOBAL_ALL); + + std::string program_data_path = get_path_relative_to_home(".local/share/checks_parser"); + create_directories(program_data_path); + + // std::string settings_file_path = + // get_path_relative_to_home(".local/share/checks_parser/settings.json"); + + // Settings s(settings_file_path); Net n; Parser p; @@ -44,22 +186,22 @@ int main(int argc, char *argv[]) { QApplication a(argc, argv); - QTranslator translator; - QString lang = "en_US"; + // QTranslator translator; + // QString lang = "en_US"; - if (s.get_all_settings().contains("language")) { - lang = QString::fromStdString(s.get_all_settings()["language"]); - } else if (translator.load(":/translation/"+QLocale::system().name()+".qm")) { - lang = QLocale::system().name(); - } else { - lang = QString::fromStdString("en_US"); - } + // if (s.get_all_settings().contains("language")) { + // lang = QString::fromStdString(s.get_all_settings()["language"]); + // } else if (translator.load(":/translation/"+QLocale::system().name()+".qm")) { + // lang = QLocale::system().name(); + // } else { + // lang = QString::fromStdString("en_US"); + // } - std::cout << "Using locale: " << lang.toStdString() << std::endl; + // std::cout << "Using locale: " << lang.toStdString() << std::endl; - translator.load(":/translation/" + lang + ".qm"); + // translator.load(":/translation/" + lang + ".qm"); - a.installTranslator(&translator); + // a.installTranslator(&translator); MainWindow w; w.update(); w.show(); diff --git a/mainwindow.cpp b/mainwindow.cpp index 59474c7..177df99 100644 --- a/mainwindow.cpp +++ b/mainwindow.cpp @@ -1,219 +1,15 @@ #include "mainwindow.h" -#include "./ui_mainwindow.h" -#include "check/check.h" -#include "exceptions/ofdrequestexception.h" -#include "goods/goods.h" -#include "outputdialog.h" -#include "adjustpicturedialog.h" -#include "settingsdialog.h" -#include "solvecaptchadialog.h" -#include -#include -#include "image/checkimage.h" -#include "utils/utils.h" -#include -#include -#include -#include -#include +#include "ui_mainwindow.h" +#include MainWindow::MainWindow(QWidget *parent) - : QMainWindow(parent), ui(new Ui::MainWindow) { + : QWidget(parent) + , ui(new Ui::MainWindow) { ui->setupUi(this); - this->setupStoresList(); + + std::cout << "test" << std::endl; } -MainWindow::~MainWindow() { delete ui; } - -void MainWindow::setupStoresList() { - parser = *(new Parser()); - - std::vector modules_names = parser.search_modules(); - - for (std::string name : modules_names) { - StoreModule m(name); - std::wstring module_name = m.get_name(); - - QString s = QString::fromStdWString(module_name); - ui->storeType->addItem(s); - } +MainWindow::~MainWindow() { + delete ui; } - -std::string MainWindow::makeRequestToOfd(std::string captcha) { - std::string checkContent = Net().fetch_check_data_from_ofdru( - ui->fn_edit->text().toStdString(), - ui->fd_edit->text().toStdString(), - ui->fi_edit->text().toStdString(), - ui->dateTimeEdit->dateTime().toString(Qt::ISODate).toStdString(), - ui->fundIncomeCombo->currentIndex() + 1, - // In the request to ofd.ru, total is in a strange format, like a string of a format where 2 last digits represent decimal part of a number. - ui->total_edit->text().toDouble() * 100, - captcha); - - return checkContent; -} - -void MainWindow::on_parseButton_clicked() { - QString s; - switch (ui->tabWidget->currentIndex()) { - case 0: - s = ui->checkContent->toPlainText(); - break; - case 1: - s = ui->checkContentFromImage->toPlainText(); - break; - case 2: - Net().get_captcha_from_ofdru(); - - std::string solved_captcha = ""; - bool success = true; - bool is_captcha_solved = true; - - do { - SolveCaptchaDialog dialog = SolveCaptchaDialog(this, &solved_captcha); - dialog.exec(); - is_captcha_solved = true; - - try { - std::string check_content = makeRequestToOfd(solved_captcha); - check = parseOfdRuAnswer(check_content); - } catch(OfdRequestException e) { - success = false; - if (!strcmp(e.what(), "Incorrect captcha")) { - is_captcha_solved = false; - QMessageBox infoDialog; - infoDialog.setText(tr("Captcha was not solved correctly!")); - infoDialog.setIcon(QMessageBox::Critical); - infoDialog.setWindowTitle(tr("Captcha is incorrect")); - infoDialog.exec(); - break; - } else if (!strcmp(e.what(), "Internal server error")) { - QMessageBox infoDialog; - infoDialog.setText(tr("Internal server error. Please, try again later.")); - infoDialog.setIcon(QMessageBox::Critical); - infoDialog.setWindowTitle(tr("Internal server error")); - infoDialog.exec(); - return; - } else if (!strcmp(e.what(), "Does not exist")) { - QMessageBox infoDialog; - infoDialog.setText(tr("Check not found. Please, ensure correctness of entered data.")); - infoDialog.setIcon(QMessageBox::Critical); - infoDialog.setWindowTitle(tr("Check was not found")); - infoDialog.exec(); - return; - } - } - } while (!is_captcha_solved); - - if (success) { - OutputDialog d = OutputDialog(this, check); - d.exec(); - } - - return; - } - - std::wstring check_plaintext = s.toStdWString(); - parser.set_module(parser.search_modules()[0]); - - std::vector c = parser.parse(check_plaintext); - - if (c.size() == 0) { - QMessageBox infoDialog; - infoDialog.setText(tr("An error has occured. Check was matched incorrectly. Vector sizes are different. Please, contact the developer.")); - infoDialog.setIcon(QMessageBox::Critical); - infoDialog.setWindowTitle(tr("Error in parsing")); - infoDialog.exec(); - return; - } - - for (auto& g : c) { - check.add_goods(g); - } - - OutputDialog d = OutputDialog(this, check); - d.show(); - d.exec(); -} - -void MainWindow::on_storeType_currentIndexChanged(int index) { - std::string module = parser.search_modules()[index]; - parser.set_module(module); -} - - -void MainWindow::on_preferencesButton_clicked() { - SettingsDialog s = SettingsDialog(); - s.show(); - s.exec(); -} - -void MainWindow::on_chooseImageButton_ofd_clicked() { - QString filename = QFileDialog::getOpenFileName(); - - if (filename == "") { - QMessageBox infoDialog; - infoDialog.setText(tr("Please, select a picture where QR code that contains info about check is present")); - infoDialog.setIcon(QMessageBox::Critical); - infoDialog.setWindowTitle(tr("Picture was not selected")); - infoDialog.exec(); - return; - } - - std::string new_text = "Selected: " + filename.toStdString(); - ui->pathLabel_ofd->setText(QString::fromStdString(new_text)); - - AdjustPictureDialog dialog = AdjustPictureDialog(this, filename.toStdString()); - connect(&dialog, &AdjustPictureDialog::decodedData, this, &MainWindow::onDecodedData); - dialog.exec(); - - ui->picture_ofd->setPixmap(QPixmap(filename)); - ui->picture_ofd->setScaledContents(true); -} - -void MainWindow::onDecodedData(std::string data) { - std::string delimiter = "&"; - std::vector dataSplit = split(data, delimiter); - - std::cout << data << std::endl; - - ui->fn_edit->setText(QString::fromStdString(dataSplit[2])); - ui->fd_edit->setText(QString::fromStdString(dataSplit[3])); - ui->fi_edit->setText(QString::fromStdString(dataSplit[4])); - - QString extractedDateTime = QString::fromStdString(split(dataSplit[0], "=")[1]); - QDateTime datetime = QDateTime::fromString(extractedDateTime, "yyyyMMddThhmm"); - ui->dateTimeEdit->setDateTime(datetime); - - int type = std::stoi(split(dataSplit[5], "=")[1]); - ui->fundIncomeCombo->setCurrentIndex(type - 1); - - std::string total = split(dataSplit[1], "=")[1]; - - ui->total_edit->setText(QString::fromStdString(total)); -} - -void MainWindow::on_chooseImageButton_ocr_clicked() -{ - QString filename = QFileDialog::getOpenFileName(); - - if (filename == "") { - QMessageBox infoDialog; - infoDialog.setText(tr("Please, select a picture to scan")); - infoDialog.setIcon(QMessageBox::Critical); - infoDialog.setWindowTitle(tr("Picture was not selected")); - infoDialog.exec(); - return; - } - - std::string new_text = "Selected: " + filename.toStdString(); - ui->pathLabel_ocr->setText(QString::fromStdString(new_text)); - - CheckImage i(filename.toStdString()); - std::string parsed = i.parse_text(); - - ui->picture_ocr->setPixmap(QPixmap(filename)); - ui->picture_ocr->setScaledContents(true); - ui->checkContentFromImage->setPlainText(QString::fromStdString(parsed)); -} - diff --git a/mainwindow.cpp.old b/mainwindow.cpp.old new file mode 100644 index 0000000..62e5fd9 --- /dev/null +++ b/mainwindow.cpp.old @@ -0,0 +1,219 @@ +#include "mainwindow.h" +#include "./ui_mainwindow.h" +#include "check/check.h" +#include "exceptions/ofdrequestexception.h" +#include "goods/goods.h" +#include "outputdialog.h" +#include "adjustpicturedialog.h" +#include "settingsdialog.h" +#include "solvecaptchadialog.h" +#include +#include +#include "image/checkimage.h" +#include "utils/utils.h" +#include +#include +#include +#include +#include + +MainWindow::MainWindow(QWidget *parent) + : QMainWindow(parent), ui(new Ui::MainWindow) { + ui->setupUi(this); + // this->setupStoresList(); +} + +MainWindow::~MainWindow() { delete ui; } + +// void MainWindow::setupStoresList() { +// parser = *(new Parser()); + +// std::vector modules_names = parser.search_modules(); + +// for (std::string name : modules_names) { +// StoreModule m(name); +// std::wstring module_name = m.get_name(); + +// QString s = QString::fromStdWString(module_name); +// ui->storeType->addItem(s); +// } +// } + +// std::string MainWindow::makeRequestToOfd(std::string captcha) { +// std::string checkContent = Net().fetch_check_data_from_ofdru( +// ui->fn_edit->text().toStdString(), +// ui->fd_edit->text().toStdString(), +// ui->fi_edit->text().toStdString(), +// ui->dateTimeEdit->dateTime().toString(Qt::ISODate).toStdString(), +// ui->fundIncomeCombo->currentIndex() + 1, +// // In the request to ofd.ru, total is in a strange format, like a string of a format where 2 last digits represent decimal part of a number. +// ui->total_edit->text().toDouble() * 100, +// captcha); + +// return checkContent; +// } + +// void MainWindow::on_parseButton_clicked() { +// QString s; +// switch (ui->tabWidget->currentIndex()) { +// case 0: +// s = ui->checkContent->toPlainText(); +// break; +// case 1: +// s = ui->checkContentFromImage->toPlainText(); +// break; +// case 2: +// Net().get_captcha_from_ofdru(); + +// std::string solved_captcha = ""; +// bool success = true; +// bool is_captcha_solved = true; + +// do { +// SolveCaptchaDialog dialog = SolveCaptchaDialog(this, &solved_captcha); +// dialog.exec(); +// is_captcha_solved = true; + +// try { +// std::string check_content = makeRequestToOfd(solved_captcha); +// check = parseOfdRuAnswer(check_content); +// } catch(OfdRequestException e) { +// success = false; +// if (!strcmp(e.what(), "Incorrect captcha")) { +// is_captcha_solved = false; +// QMessageBox infoDialog; +// infoDialog.setText(tr("Captcha was not solved correctly!")); +// infoDialog.setIcon(QMessageBox::Critical); +// infoDialog.setWindowTitle(tr("Captcha is incorrect")); +// infoDialog.exec(); +// break; +// } else if (!strcmp(e.what(), "Internal server error")) { +// QMessageBox infoDialog; +// infoDialog.setText(tr("Internal server error. Please, try again later.")); +// infoDialog.setIcon(QMessageBox::Critical); +// infoDialog.setWindowTitle(tr("Internal server error")); +// infoDialog.exec(); +// return; +// } else if (!strcmp(e.what(), "Does not exist")) { +// QMessageBox infoDialog; +// infoDialog.setText(tr("Check not found. Please, ensure correctness of entered data.")); +// infoDialog.setIcon(QMessageBox::Critical); +// infoDialog.setWindowTitle(tr("Check was not found")); +// infoDialog.exec(); +// return; +// } +// } +// } while (!is_captcha_solved); + +// if (success) { +// OutputDialog d = OutputDialog(this, check); +// d.exec(); +// } + +// return; +// } + +// std::wstring check_plaintext = s.toStdWString(); +// parser.set_module(parser.search_modules()[0]); + +// std::vector c = parser.parse(check_plaintext); + +// if (c.size() == 0) { +// QMessageBox infoDialog; +// infoDialog.setText(tr("An error has occured. Check was matched incorrectly. Vector sizes are different. Please, contact the developer.")); +// infoDialog.setIcon(QMessageBox::Critical); +// infoDialog.setWindowTitle(tr("Error in parsing")); +// infoDialog.exec(); +// return; +// } + +// for (auto& g : c) { +// check.add_goods(g); +// } + +// OutputDialog d = OutputDialog(this, check); +// d.show(); +// d.exec(); +// } + +// void MainWindow::on_storeType_currentIndexChanged(int index) { +// std::string module = parser.search_modules()[index]; +// parser.set_module(module); +// } + + +// void MainWindow::on_preferencesButton_clicked() { +// SettingsDialog s = SettingsDialog(); +// s.show(); +// s.exec(); +// } + +// void MainWindow::on_chooseImageButton_ofd_clicked() { +// QString filename = QFileDialog::getOpenFileName(); + +// if (filename == "") { +// QMessageBox infoDialog; +// infoDialog.setText(tr("Please, select a picture where QR code that contains info about check is present")); +// infoDialog.setIcon(QMessageBox::Critical); +// infoDialog.setWindowTitle(tr("Picture was not selected")); +// infoDialog.exec(); +// return; +// } + +// std::string new_text = "Selected: " + filename.toStdString(); +// ui->pathLabel_ofd->setText(QString::fromStdString(new_text)); + +// AdjustPictureDialog dialog = AdjustPictureDialog(this, filename.toStdString()); +// connect(&dialog, &AdjustPictureDialog::decodedData, this, &MainWindow::onDecodedData); +// dialog.exec(); + +// ui->picture_ofd->setPixmap(QPixmap(filename)); +// ui->picture_ofd->setScaledContents(true); +// } + +// void MainWindow::onDecodedData(std::string data) { +// std::string delimiter = "&"; +// std::vector dataSplit = split(data, delimiter); + +// std::cout << data << std::endl; + +// ui->fn_edit->setText(QString::fromStdString(dataSplit[2])); +// ui->fd_edit->setText(QString::fromStdString(dataSplit[3])); +// ui->fi_edit->setText(QString::fromStdString(dataSplit[4])); + +// QString extractedDateTime = QString::fromStdString(split(dataSplit[0], "=")[1]); +// QDateTime datetime = QDateTime::fromString(extractedDateTime, "yyyyMMddThhmm"); +// ui->dateTimeEdit->setDateTime(datetime); + +// int type = std::stoi(split(dataSplit[5], "=")[1]); +// ui->fundIncomeCombo->setCurrentIndex(type - 1); + +// std::string total = split(dataSplit[1], "=")[1]; + +// ui->total_edit->setText(QString::fromStdString(total)); +// } + +// void MainWindow::on_chooseImageButton_ocr_clicked() +// { +// QString filename = QFileDialog::getOpenFileName(); + +// if (filename == "") { +// QMessageBox infoDialog; +// infoDialog.setText(tr("Please, select a picture to scan")); +// infoDialog.setIcon(QMessageBox::Critical); +// infoDialog.setWindowTitle(tr("Picture was not selected")); +// infoDialog.exec(); +// return; +// } + +// std::string new_text = "Selected: " + filename.toStdString(); +// ui->pathLabel_ocr->setText(QString::fromStdString(new_text)); + +// CheckImage i(filename.toStdString()); +// std::string parsed = i.parse_text(); + +// ui->picture_ocr->setPixmap(QPixmap(filename)); +// ui->picture_ocr->setScaledContents(true); +// ui->checkContentFromImage->setPlainText(QString::fromStdString(parsed)); +// } + diff --git a/mainwindow.h b/mainwindow.h index 14386a7..84bf578 100644 --- a/mainwindow.h +++ b/mainwindow.h @@ -1,44 +1,25 @@ #ifndef MAINWINDOW_H #define MAINWINDOW_H -#include +#include +#include +#include -#include "check/check.h" -#include "parser/parser.h" - -QT_BEGIN_NAMESPACE namespace Ui { class MainWindow; } -QT_END_NAMESPACE -class MainWindow : public QMainWindow { +class MainWindow : public QWidget +{ Q_OBJECT - Check check; - Parser parser; - public: - MainWindow(QWidget *parent = nullptr); + explicit MainWindow(QWidget *parent = nullptr); ~MainWindow(); - void setupStoresList(); - - Check get_check(); - void onDecodedData(std::string); - - std::string makeRequestToOfd(std::string captcha); -private slots: - void on_parseButton_clicked(); - - void on_storeType_currentIndexChanged(int index); - - void on_preferencesButton_clicked(); - - void on_chooseImageButton_ofd_clicked(); - - void on_chooseImageButton_ocr_clicked(); - -private: Ui::MainWindow *ui; +private slots: +private: + }; + #endif // MAINWINDOW_H diff --git a/mainwindow.h.old b/mainwindow.h.old new file mode 100644 index 0000000..ed77ada --- /dev/null +++ b/mainwindow.h.old @@ -0,0 +1,44 @@ +#ifndef MAINWINDOW_H +#define MAINWINDOW_H + +#include + +#include "check/check.h" +#include "parser/parser.h" + +QT_BEGIN_NAMESPACE +namespace Ui { +class MainWindow; +} +QT_END_NAMESPACE + +class MainWindow : public QMainWindow { + Q_OBJECT + + Check check; + Parser parser; + +public: + MainWindow(QWidget *parent = nullptr); + ~MainWindow(); + // void setupStoresList(); + + // Check get_check(); + // void onDecodedData(std::string); + + // std::string makeRequestToOfd(std::string captcha); +private slots: + // void on_parseButton_clicked(); + + // void on_storeType_currentIndexChanged(int index); + + // void on_preferencesButton_clicked(); + + // void on_chooseImageButton_ofd_clicked(); + + // void on_chooseImageButton_ocr_clicked(); + +private: + Ui::MainWindow *ui; +}; +#endif // MAINWINDOW_H diff --git a/media.qrc b/media.qrc new file mode 100644 index 0000000..99e4a07 --- /dev/null +++ b/media.qrc @@ -0,0 +1,7 @@ + + + assets/icons/email-text.svg + assets/icons/OCR.svg + assets/icons/OFD.svg + + diff --git a/ocrscene.cpp b/ocrscene.cpp new file mode 100644 index 0000000..dc1e782 --- /dev/null +++ b/ocrscene.cpp @@ -0,0 +1,73 @@ +#include "ocrscene.h" +#include "ui_ocrscene.h" + +#include +#include +#include + +#include + +#include + +OCRScene::OCRScene(QWidget *parent) + : QWidget(parent) + , ui(new Ui::OCRScene) { + ui->setupUi(this); + + auto modules = parser.get_modules_names(); + + for (auto &module : modules) { + ui->store_combo_box->addItem(QString::fromStdString(module)); + } +} + +OCRScene::~OCRScene() { + delete ui; +} + +void OCRScene::on_parse_button_clicked() { + std::wstring checkContent = ui->check_text_edit->toPlainText().toStdWString(); + + parser.set_module(parser.search_modules()[ui->store_combo_box->currentIndex()]); + + std::vector goods = parser.parse(checkContent); + if (goods.size() == 0) { + QMessageBox infoDialog; + infoDialog.setText(tr("An error has occured. Check was matched incorrectly. Vector sizes are different. Please, contact the developer.")); + infoDialog.setIcon(QMessageBox::Critical); + infoDialog.setWindowTitle(tr("Error in parsing")); + infoDialog.exec(); + return; + } + + Check check; + check.add_goods(goods); + + OutputDialog d(this, check); + d.show(); + d.exec(); +} + +void OCRScene::on_choose_image_button_clicked() { + QString filename = QFileDialog::getOpenFileName(); + + if (filename == "") { + QMessageBox infoDialog; + infoDialog.setText(tr("Please, select a picture to scan")); + infoDialog.setIcon(QMessageBox::Critical); + infoDialog.setWindowTitle(tr("Picture was not selected")); + infoDialog.exec(); + return; + } + + std::string new_text = "Selected: " + filename.toStdString(); + ui->path_to_image_label->setText(tr("Path to image: ")+ filename); + + CheckImage i(filename.toStdString()); + std::string parsed = i.parse_text(); + + ui->check_text_edit->setPlainText(QString::fromStdString(parsed)); + + +} + diff --git a/ocrscene.h b/ocrscene.h new file mode 100644 index 0000000..2909825 --- /dev/null +++ b/ocrscene.h @@ -0,0 +1,29 @@ +#ifndef OCRSCENE_H +#define OCRSCENE_H + +#include "parser/parser.h" +#include + +namespace Ui { +class OCRScene; +} + +class OCRScene : public QWidget +{ + Q_OBJECT + +public: + explicit OCRScene(QWidget *parent = nullptr); + ~OCRScene(); + +private slots: + void on_parse_button_clicked(); + + void on_choose_image_button_clicked(); + +private: + Ui::OCRScene *ui; + Parser parser; +}; + +#endif // OCRSCENE_H diff --git a/ofdscene.cpp b/ofdscene.cpp new file mode 100644 index 0000000..fad6d35 --- /dev/null +++ b/ofdscene.cpp @@ -0,0 +1,150 @@ +#include "ofdscene.h" +#include "ui_ofdscene.h" +#include "utils/utils.h" + +#include +#include +#include +#include +#include +#include + +#include + +#include + +OFDScene::OFDScene(QWidget *parent) + : QWidget(parent) + , ui(new Ui::OFDScene) { + ui->setupUi(this); +} + +OFDScene::~OFDScene() { + delete ui; +} + +void OFDScene::startHttpServer() { + std::string localIp = ""; + try { + localIp = get_local_ip_address(); + } catch(std::exception e) { + std::cerr << e.what() << std::endl; + return; + } + + httplib::Server svr; + //TODO: generate random port from 1024 to 65535 and check if its used. + svr.Get("/", [&](const httplib::Request &, httplib::Response &res){ + res.set_redirect("http://"+ localIp +":8080/", 301); + }); + + svr.listen("0.0.0.0", 8080); +} + +void OFDScene::on_choose_image_button_clicked() { + QString filename = QFileDialog::getOpenFileName(); + + if (filename == "") { + QMessageBox infoDialog; + infoDialog.setText(tr("Please, select a picture where QR code that contains info about check is present")); + infoDialog.setIcon(QMessageBox::Critical); + infoDialog.setWindowTitle(tr("Picture was not selected")); + infoDialog.exec(); + return; + } + + + ui->info_label->setText(tr("Selected image: ") + filename); + + AdjustPictureDialog dialog = AdjustPictureDialog(this, filename.toStdString()); + connect(&dialog, &AdjustPictureDialog::decodedData, this, &OFDScene::onDataDecode); + dialog.exec(); +} + +void OFDScene::onDataDecode(std::string data) { + std::vector dataSplit = split(data, "&"); + + ui->fn_line_edit->setText(QString::fromStdString(split(dataSplit[2], "=")[1])); + ui->fd_line_edit->setText(QString::fromStdString(split(dataSplit[3], "=")[1])); + ui->fi_line_edit->setText(QString::fromStdString(split(dataSplit[4], "=")[1])); + + QString extractedDateTime = QString::fromStdString(split(dataSplit[0], "=")[1]); + //TODO: some QRs contain datetime in format yyyyMMddThhmmss. Perhaps there is more different formats, should write function to detect them. + QDateTime datetime = QDateTime::fromString(extractedDateTime, "yyyyMMddThhmm"); + ui->purchase_datetime_edit->setDateTime(datetime); + + int type = std::stoi(split(dataSplit[5], "=")[1]); + ui->operation_type_combo_box->setCurrentIndex(type - 1); + + std::string total = split(dataSplit[1], "=")[1]; + + ui->total_spin_box->setValue(std::stod(total)); +} + + +void OFDScene::on_parse_button_clicked() { + Net net; + net.get_captcha_from_ofdru(); + + std::string solved_captcha = ""; + bool success = true; + bool is_captcha_solved = true; + Check check; + + do { + SolveCaptchaDialog dialog = SolveCaptchaDialog(this, &solved_captcha); + dialog.exec(); + is_captcha_solved = true; + + try { + std::string check_content = net.fetch_check_data_from_ofdru( + ui->fn_line_edit->text().toStdString(), + ui->fd_line_edit->text().toStdString(), + ui->fi_line_edit->text().toStdString(), + ui->purchase_datetime_edit->dateTime().toString(Qt::ISODate).toStdString(), + ui->operation_type_combo_box->currentIndex() + 1, + // In the request to ofd.ru, total is in a format with 2 last digits represent decimal part of a number. + ui->total_spin_box->text().toDouble() * 100, + solved_captcha); + + check = parseOfdRuAnswer(check_content); + } catch(OfdRequestException e) { + success = false; + if (!strcmp(e.what(), "Incorrect captcha")) { + is_captcha_solved = false; + QMessageBox infoDialog; + infoDialog.setText(tr("Captcha was not solved correctly!")); + infoDialog.setIcon(QMessageBox::Critical); + infoDialog.setWindowTitle(tr("Captcha is incorrect")); + infoDialog.exec(); + break; + } else if (!strcmp(e.what(), "Internal server error")) { + QMessageBox infoDialog; + infoDialog.setText(tr("Internal server error. Please, try again later.")); + infoDialog.setIcon(QMessageBox::Critical); + infoDialog.setWindowTitle(tr("Internal server error")); + infoDialog.exec(); + return; + } else if (!strcmp(e.what(), "Does not exist")) { + QMessageBox infoDialog; + infoDialog.setText(tr("Check not found. Please, ensure correctness of entered data.")); + infoDialog.setIcon(QMessageBox::Critical); + infoDialog.setWindowTitle(tr("Check was not found")); + infoDialog.exec(); + return; + } + } + } while (!is_captcha_solved); + + if (success) { + OutputDialog d = OutputDialog(this, check); + d.exec(); + } + +} + + +void OFDScene::on_binary_eye_button_clicked() { + +} + diff --git a/ofdscene.h b/ofdscene.h new file mode 100644 index 0000000..83901a3 --- /dev/null +++ b/ofdscene.h @@ -0,0 +1,30 @@ +#ifndef OFDSCENE_H +#define OFDSCENE_H + +#include + +namespace Ui { +class OFDScene; +} + +class OFDScene : public QWidget +{ + Q_OBJECT + +public: + explicit OFDScene(QWidget *parent = nullptr); + ~OFDScene(); + void startHttpServer(); +private slots: + void on_choose_image_button_clicked(); + void onDataDecode(std::string data); + + void on_parse_button_clicked(); + + void on_binary_eye_button_clicked(); + +private: + Ui::OFDScene *ui; +}; + +#endif // OFDSCENE_H diff --git a/mainwindow.ui b/old_mainwindow.ui similarity index 99% rename from mainwindow.ui rename to old_mainwindow.ui index f4ebd1e..db5c247 100644 --- a/mainwindow.ui +++ b/old_mainwindow.ui @@ -73,7 +73,7 @@ - 2 + 1 @@ -384,7 +384,7 @@ 0 0 817 - 23 + 33 diff --git a/outputdialog.ui b/outputdialog.ui deleted file mode 100644 index c6e462f..0000000 --- a/outputdialog.ui +++ /dev/null @@ -1,215 +0,0 @@ - - - OutputDialog - - - - 0 - 0 - 586 - 431 - - - - Dialog - - - - - 410 - 390 - 166 - 26 - - - - Qt::Horizontal - - - QDialogButtonBox::Cancel|QDialogButtonBox::Ok - - - - - - 10 - 20 - 271 - 18 - - - - Path to export: - - - - - - 290 - 20 - 80 - 26 - - - - Choose - - - - - - 10 - 50 - 371 - 24 - - - - Print header - - - - - - 10 - 130 - 401 - 221 - - - - - Goods name - - - - - Goods price - - - - - Goods quantity - - - - - Goods net weight - - - - - Goods total - - - - - position - - - - - name - - - - - 1 - - - - - Name - - - - - 2 - - - - - Price - - - - - 3 - - - - - Quantity - - - - - 4 - - - - - Net weight - - - - - 5 - - - - - Total price - - - - - - - 10 - 90 - 381 - 24 - - - - Print total - - - - - - - buttonBox - accepted() - OutputDialog - accept() - - - 248 - 254 - - - 157 - 274 - - - - - buttonBox - rejected() - OutputDialog - reject() - - - 316 - 260 - - - 286 - 274 - - - - - diff --git a/parser/module.cpp b/parser/module.cpp index ebdc002..dba8d44 100644 --- a/parser/module.cpp +++ b/parser/module.cpp @@ -31,7 +31,7 @@ StoreModule::StoreModule(std::string path) { std::vector StoreModule::parse_name(std::wstring str) { std::vector result; - std::wregex r(this->goods_name_regex, std::regex::collate); + std::wregex r(this->goods_name_regex, std::regex_constants::multiline); for (std::wsregex_iterator it{str.begin(), str.end(), r}, end{}; it != end; it++) { diff --git a/parser/parser.cpp b/parser/parser.cpp index a35a2b1..d15a0ff 100644 --- a/parser/parser.cpp +++ b/parser/parser.cpp @@ -4,6 +4,7 @@ #include "../settings/settings.h" #include "../utils/utils.h" #include +#include #if __GNUC__ < 8 && __clang_major__ < 17 # include @@ -14,6 +15,27 @@ using namespace std::filesystem; #endif +static void dumpVectorsToStdErr(std::vector &goods_names, std::vector &goods_prices, std::vector& goods_quantities) { + std::cerr << goods_names.size() << " " << goods_prices.size() << " " << goods_quantities.size() << std::endl; + std::cerr << "Found goods names: "; + for (auto &goods_name : goods_names) { + std::cerr << goods_name << " "; + } + std::cerr << std::endl; + + std::cerr << "Found goods prices: "; + for (auto &goods_price : goods_prices) { + std::cerr << goods_price << " "; + } + std::cerr << std::endl; + + std::cerr << "Found goods names: "; + for (auto &goods_quantity : goods_quantities) { + std::cerr << goods_quantity << " "; + } + std::cerr << std::endl; +} + Parser::Parser() {} std::vector Parser::search_modules() { @@ -31,13 +53,28 @@ std::vector Parser::search_modules() { std::vector modules_files; - for (auto file : directory_iterator(path)) { + for (auto &file : directory_iterator(path)) { modules_files.push_back(file.path()); } return modules_files; } +std::vector Parser::get_modules_names() { + std::vector modules = this->search_modules(); + std::vector names = {}; + + for (std::string &modulePath : modules) { + std::ifstream inputFile(modulePath); + + nlohmann::json module = nlohmann::json::parse(inputFile); + std::string name = module["name"]; + names.push_back(name); + } + + return names; +} + void Parser::set_module(std::string path) { module = StoreModule(path); } std::vector Parser::parse(std::wstring check_plaintext) { @@ -53,6 +90,9 @@ std::vector Parser::parse(std::wstring check_plaintext) { if (goods_names.size() != goods_prices.size() || goods_names.size() != goods_quantities.size() || goods_prices.size() != goods_quantities.size()) { + + dumpVectorsToStdErr(goods_names, goods_prices, goods_quantities); + //Error. Amount of names, prices or quantities are not equal. That means, that some regex(es) has mismatched. return {}; } diff --git a/parser/parser.h b/parser/parser.h index 5e4136f..c845785 100644 --- a/parser/parser.h +++ b/parser/parser.h @@ -15,6 +15,8 @@ public: std::vector search_modules(); + std::vector get_modules_names(); + std::vector check_updates(); void set_module(std::string); diff --git a/scenes.qrc b/scenes.qrc new file mode 100644 index 0000000..4704a99 --- /dev/null +++ b/scenes.qrc @@ -0,0 +1,10 @@ + + + scenes/outputdialog.ui + scenes/emailtextscene.ui + scenes/ocrscene.ui + scenes/mainwindow.ui + scenes/ofdscene.ui + scenes/solvecaptchadialog.ui + + diff --git a/adjustpicturedialog.ui b/scenes/adjustpicturedialog.ui similarity index 100% rename from adjustpicturedialog.ui rename to scenes/adjustpicturedialog.ui diff --git a/scenes/emailtextscene.ui b/scenes/emailtextscene.ui new file mode 100644 index 0000000..84573c2 --- /dev/null +++ b/scenes/emailtextscene.ui @@ -0,0 +1,80 @@ + + + EmailTextScene + + + + 0 + 0 + 1077 + 608 + + + + Form + + + + + + + 0 + 0 + + + + Check content + + + + + + + Parse + + + + + + + + 0 + 0 + + + + Store: + + + + + + + + + + + 0 + 0 + + + + Back + + + + + + + + 0 + 0 + + + + + + + + + diff --git a/scenes/mainwindow.ui b/scenes/mainwindow.ui new file mode 100644 index 0000000..11127a6 --- /dev/null +++ b/scenes/mainwindow.ui @@ -0,0 +1,121 @@ + + + MainWindow + + + + 0 + 0 + 1039 + 693 + + + + + 0 + 0 + + + + + 971 + 0 + + + + Form + + + + + + 6 + + + + + Optical Character Recognition + + + + + + + :/icons/assets/icons/OCR.svg:/icons/assets/icons/OCR.svg + + + + 128 + 128 + + + + + + + + + 0 + 0 + + + + Text from E-Mail + + + false + + + + + + + :/icons/assets/icons/email-text.svg:/icons/assets/icons/email-text.svg + + + + 128 + 128 + + + + false + + + false + + + false + + + false + + + false + + + + + + + + + + + :/icons/assets/icons/OFD.svg:/icons/assets/icons/OFD.svg + + + + 128 + 128 + + + + + + + + + + + diff --git a/scenes/ocrscene.ui b/scenes/ocrscene.ui new file mode 100644 index 0000000..1c9aa59 --- /dev/null +++ b/scenes/ocrscene.ui @@ -0,0 +1,127 @@ + + + OCRScene + + + + 0 + 0 + 992 + 634 + + + + + 0 + 0 + + + + Form + + + + 8 + + + + + + 0 + 0 + + + + + + + + + 0 + 0 + + + + Parse + + + + + + + + 0 + 0 + + + + + 0 + 0 + + + + Choose + + + + + + + + 0 + 0 + + + + Back + + + + + + + + 0 + 0 + + + + Recognized text will be shown below as soon as image will be processed. Please, edit it + + + + + + + + 0 + 0 + + + + Path to image: + + + + + + + + + + + 0 + 0 + + + + Store: + + + + + + + + diff --git a/scenes/ofdscene.ui b/scenes/ofdscene.ui new file mode 100644 index 0000000..4e44026 --- /dev/null +++ b/scenes/ofdscene.ui @@ -0,0 +1,194 @@ + + + OFDScene + + + + 0 + 0 + 894 + 625 + + + + Form + + + + QLayout::SizeConstraint::SetDefaultConstraint + + + + + Total + + + + + + + + 0 + 0 + + + + Back + + + + + + + + 0 + 0 + + + + or + + + + + + + 4294967296.000000000000000 + + + + + + + + 0 + 0 + + + + FD (Fiscal Document) + + + + + + + Date and time of purchase + + + + + + + + Funds income + + + + + Funds return + + + + + Funds spend + + + + + Spends return + + + + + + + + + + + + 0 + 0 + + + + + + + + + + + + 0 + 0 + + + + + + + + + + + Use your phone as a QR code scanner + + + + + + + + 0 + 0 + + + + FN (Fiscal Number) + + + + + + + FI (Fiscal Identifier) + + + + + + + + + + Choose image on your PC + + + + + + + Operation type + + + + + + + + 0 + 0 + + + + Parse + + + + + + + + diff --git a/scenes/outputdialog.ui b/scenes/outputdialog.ui new file mode 100644 index 0000000..f458fdf --- /dev/null +++ b/scenes/outputdialog.ui @@ -0,0 +1,187 @@ + + + OutputDialog + + + + 0 + 0 + 892 + 537 + + + + Dialog + + + + + + Path to export: + + + + + + + Choose + + + + + + + Print header + + + + + + + Print total + + + + + + + + 0 + 0 + + + + + Goods name + + + + + Goods price + + + + + Goods quantity + + + + + Goods net weight + + + + + Goods total + + + + + position + + + + + name + + + + + 1 + + + + + Name + + + + + 2 + + + + + Price + + + + + 3 + + + + + Quantity + + + + + 4 + + + + + Net weight + + + + + 5 + + + + + Total price + + + + + + + + Qt::Orientation::Horizontal + + + QDialogButtonBox::StandardButton::Cancel|QDialogButtonBox::StandardButton::Ok + + + + + + + + + buttonBox + accepted() + OutputDialog + accept() + + + 248 + 254 + + + 157 + 274 + + + + + buttonBox + rejected() + OutputDialog + reject() + + + 316 + 260 + + + 286 + 274 + + + + + diff --git a/solvecaptchadialog.ui b/scenes/solvecaptchadialog.ui similarity index 100% rename from solvecaptchadialog.ui rename to scenes/solvecaptchadialog.ui diff --git a/translations/en_US.ts b/translations/en_US.ts index 8e79468..49b8dfd 100644 --- a/translations/en_US.ts +++ b/translations/en_US.ts @@ -4,12 +4,12 @@ AdjustPictureDialog - + Dialog Dialog - + Please, zoom to qr code and adjust contrast so that qr code looks sharp Please, zoom to qr code and adjust contrast so that qr code looks sharp @@ -25,295 +25,502 @@ - MainWindow + EmailTextScene - - MainWindow - Главное окно + + Form + Form - Store type - Store type + Store type - - Parse - Parse - - - - Preferences - Preferences - - - - Text - Text - - - + Check content Check content - - OCR - OCR = Optical Character Recognition - OCR + + Parse + Parse - - - Choose - Choose + + Store: + Store: - - - Path to image: - Path to image: + + Back + Back - - Here is recognised check text. Please, edit it if something's wrong: - Here is recognised check text. Please, edit it if something's wrong: - - - - OFD - OFD = Оператор Фискальных Данных - OFD - - - - 0000000000000000 - 0000000000000000 - - - - FN (Fiscal Number) - FN = Фискальный Номер - FN (Fiscal Number) - - - - FD (Fiscal Document) - FD = Фискальный Документ - FD (Fiscal Document) - - - - - 0000000000 - 000000000 - - - - FI (Fiscal Identifier) - FI = Фискальный Признак - FI (Fiscal Identifier) - - - - Funds income - Приход средств - Funds income - - - - Funds return - Возврат средств - Funds return - - - - Funds spend - Расход средств - Funds spend - - - - Spends return - Возврат расхода - Spends return - - - - Total - Total - - - - checks parser - checks parser - - - - Captcha was not solved correctly! - Captcha was not solved correctly! - - - - Captcha is incorrect - Captcha is incorrect - - - - Internal server error. Please, try again later. - Internal server error. Please, try again later. - - - - Internal server error - Internal server error - - - - Check not found. Please, ensure correctness of entered data. - Check not found. Please, ensure correctness of entered data. - - - - Check was not found - - - - + An error has occured. Check was matched incorrectly. Vector sizes are different. Please, contact the developer. An error has occured. Check was matched incorrectly. Vector sizes are different. Please, contact the developer. - + + Error in parsing + Error in parsing + + + + MainWindow + + MainWindow + Главное окно + + + Store type + Store type + + + Parse + Parse + + + Preferences + Preferences + + + Text + Text + + + Check content + Check content + + + OCR + OCR = Optical Character Recognition + OCR + + + Choose + Choose + + + Path to image: + Path to image: + + + Here is recognised check text. Please, edit it if something's wrong: + Here is recognised check text. Please, edit it if something's wrong: + + + OFD + OFD = Оператор Фискальных Данных + OFD + + + 0000000000000000 + 0000000000000000 + + + FN (Fiscal Number) + FN = Фискальный Номер + FN (Fiscal Number) + + + FD (Fiscal Document) + FD = Фискальный Документ + FD (Fiscal Document) + + + 0000000000 + 000000000 + + + FI (Fiscal Identifier) + FI = Фискальный Признак + FI (Fiscal Identifier) + + + Funds income + Приход средств + Funds income + + + Funds return + Возврат средств + Funds return + + + Funds spend + Расход средств + Funds spend + + + Spends return + Возврат расхода + Spends return + + + Total + Total + + + checks parser + checks parser + + + Captcha was not solved correctly! + Captcha was not solved correctly! + + + Captcha is incorrect + Captcha is incorrect + + + Internal server error. Please, try again later. + Internal server error. Please, try again later. + + + Internal server error + Internal server error + + + Check not found. Please, ensure correctness of entered data. + Check not found. Please, ensure correctness of entered data. + + + An error has occured. Check was matched incorrectly. Vector sizes are different. Please, contact the developer. + An error has occured. Check was matched incorrectly. Vector sizes are different. Please, contact the developer. + + + Error in parsing + Error in parsing + + + Please, select a picture where QR code that contains info about check is present + Please, select a picture where QR code that contains info about check is present + + + Picture was not selected + Picture was not selected + + + Please, select a picture to scan + Please, select a picture to scan + + + + Form + Form + + + + Optical Character Recognition + Optical Character Recognition + + + + Text from E-Mail + Text from E-Mail + + + + OCRScene + + + Form + Form + + + + Choose + Choose + + + + Path to image: + Path to image: + + + + Store: + Store: + + + + Recognized text will be shown below as soon as image will be processed. Please, edit it + Recognized text will be shown below as soon as image will be processed. Please, edit it + + + + Back + Back + + + + Parse + Parse + + + + An error has occured. Check was matched incorrectly. Vector sizes are different. Please, contact the developer. + An error has occured. Check was matched incorrectly. Vector sizes are different. Please, contact the developer. + + + Error in parsing Error in parsing - - Please, select a picture where QR code that contains info about check is present - Please, select a picture where QR code that contains info about check is present + + Please, select a picture to scan + Please, select a picture to scan - - + Picture was not selected Picture was not selected - - Please, select a picture to scan - Please, select a picture to scan + + Path to image: + Path to image: + + + + OFDScene + + + Form + Form + + + + Total + Total + + + + Back + Back + + + + or + or + + + + FD (Fiscal Document) + FD (Fiscal Document) + + + + Date and time of purchase + Date and time of purchase + + + + Funds income + Funds income + + + + Funds return + Funds return + + + + Funds spend + Funds spend + + + + Spends return + Spends return + + + + Use your phone as a QR code scanner + Use your phone as a QR code scanner + + + + FN (Fiscal Number) + FN (Fiscal Number) + + + + FI (Fiscal Identifier) + FI (Fiscal Identifier) + + + + Choose image on your PC + Choose image on your PC + + + + Operation type + Operation type + + + + Parse + Parse + + + + Please, select a picture where QR code that contains info about check is present + Please, select a picture where QR code that contains info about check is present + + + + Picture was not selected + Picture was not selected + + + + Selected image: + Selected image: + + + + Captcha was not solved correctly! + Captcha was not solved correctly! + + + + Captcha is incorrect + Captcha is incorrect + + + + Internal server error. Please, try again later. + Internal server error. Please, try again later. + + + + Internal server error + Internal server error + + + + Check not found. Please, ensure correctness of entered data. + Check not found. Please, ensure correctness of entered data. + + + + Check was not found + Check was not found OutputDialog - + Dialog Dialog - + Path to export: Path to export: - + Choose Choose - + Print header Print header - + Goods name Goods name - + Goods price Goods price - + Goods quantity Goods quality - + Goods net weight Goods net weight - + Goods total Goods total - + position position - + name name - + 1 1 - + Name Name - + 2 2 - + Price Price - + 3 3 - + Quantity Quantity - + 4 4 - + Net weight Net Weight - + 5 5 - + Total price Total price - + Print total Print total @@ -321,20 +528,18 @@ SettingsDialog - You need to restart program to apply language changes - You need to restart program to apply language changes + You need to restart program to apply language changes - Restart required - Restart required + Restart required SolveCaptchaDialog - + Dialog Dialog @@ -352,110 +557,88 @@ settingsdialog - Dialog - Dialog + Dialog - Goods name position - Goods name position + Goods name position - Goods price per unit alias - Goods price per unit alias + Goods price per unit alias - TextLabel - Language + Language - en_US - en_US + en_US - ru_RU - ru_RU + ru_RU - - Choose - Choose + Choose - Print header - Print header + Print header - Goods net weight alias - Goods net weight alias + Goods net weight alias - Stores modules url - Stores modules url + Stores modules url - Goods total alias - Goods total alias + Goods total alias - Goods name alias - Goods name alias + Goods name alias - Goods quantity alias - Goods quantity alias + Goods quantity alias - Stores modules directory - Stores modules directory + Stores modules directory - OFD modules directory - OFD modules directory + OFD modules directory - Goods price per unit position - Goods price per unit position + Goods price per unit position - Goods net weight position - Goods net weight position + Goods net weight position - OFD modules url - OFD modules url + OFD modules url - Goods total position - Goods total position + Goods total position - Goods quantity position - Goods quantity position + Goods quantity position - Print total - Print total + Print total diff --git a/translations/ru_RU.ts b/translations/ru_RU.ts index a2bf595..de475ae 100644 --- a/translations/ru_RU.ts +++ b/translations/ru_RU.ts @@ -4,12 +4,12 @@ AdjustPictureDialog - + Dialog Диалог - + Please, zoom to qr code and adjust contrast so that qr code looks sharp Пожалуйста, приблизьте QR код и настройте контраст, чтобы он читался @@ -25,291 +25,502 @@ - MainWindow + EmailTextScene - - MainWindow - ГлавноеОкно + + Form + Форма - Store type - Магазин + Магазин - - Parse - Парсить - - - - Preferences - Настройки - - - - Text - Текст - - - + Check content Контент чека - - OCR - Оптическое Распознавание Символов - ОРС + + Parse + Парсить - - - Choose - Выбрать + + Store: + Магазин: - - - Path to image: - Путь к изображению: + + Back + Назад - - Here is recognised check text. Please, edit it if something's wrong: - Ниже приведён распознанный текст. Пожалуйста, отредактируйте его: - - - - OFD - Оператор Фискальных Данных - ОФД - - - - 0000000000000000 - 0000000000000000 - - - - FN (Fiscal Number) - Фискальный Норма - ФН - - - - FD (Fiscal Document) - Фискальный Документ - ФД - - - - - 0000000000 - 000000000 - - - - FI (Fiscal Identifier) - Фискальный Признак - ФП - - - - Funds income - Приход средств - - - - Funds return - Возврат средств - - - - Funds spend - Расход средств - - - - Spends return - Возврат расхода - - - - Total - Итого - - - - checks parser - Парсер чеков - - - - Captcha was not solved correctly! - Капча была решена неверно! - - - - Captcha is incorrect - Капча введена неверно - - - - Internal server error. Please, try again later. - Внутренняя ошибка сервера. Пожалуйста, попробуйте снова позже. - - - - Internal server error - Внутренняя ошибка сервера - - - - Check not found. Please, ensure correctness of entered data. - Чек не найден. Пожалуйста, убедитесь в правильности введённых данных. - - - - Check was not found - Чек не найден - - - + An error has occured. Check was matched incorrectly. Vector sizes are different. Please, contact the developer. Произошла ошибка. Чек был прочитан неверно. Размеры векторов различаются. Пожалуйста, сообщите об этом разработчику. - + + Error in parsing + Ошибка в парсинге + + + + MainWindow + + MainWindow + ГлавноеОкно + + + Store type + Магазин + + + Parse + Парсить + + + Preferences + Настройки + + + Text + Текст + + + Check content + Контент чека + + + OCR + Оптическое Распознавание Символов + ОРС + + + Choose + Выбрать + + + Path to image: + Путь к изображению: + + + Here is recognised check text. Please, edit it if something's wrong: + Ниже приведён распознанный текст. Пожалуйста, отредактируйте его: + + + OFD + Оператор Фискальных Данных + ОФД + + + 0000000000000000 + 0000000000000000 + + + FN (Fiscal Number) + Фискальный Норма + ФН + + + FD (Fiscal Document) + Фискальный Документ + ФД + + + 0000000000 + 000000000 + + + FI (Fiscal Identifier) + Фискальный Признак + ФП + + + Funds income + Приход средств + + + Funds return + Возврат средств + + + Funds spend + Расход средств + + + Spends return + Возврат расхода + + + Total + Итого + + + checks parser + Парсер чеков + + + Captcha was not solved correctly! + Капча была решена неверно! + + + Captcha is incorrect + Капча введена неверно + + + Internal server error. Please, try again later. + Внутренняя ошибка сервера. Пожалуйста, попробуйте снова позже. + + + Internal server error + Внутренняя ошибка сервера + + + Check not found. Please, ensure correctness of entered data. + Чек не найден. Пожалуйста, убедитесь в правильности введённых данных. + + + Check was not found + Чек не найден + + + An error has occured. Check was matched incorrectly. Vector sizes are different. Please, contact the developer. + Произошла ошибка. Чек был прочитан неверно. Размеры векторов различаются. Пожалуйста, сообщите об этом разработчику. + + + Error in parsing + Ошибка в парсинге + + + Please, select a picture where QR code that contains info about check is present + Пожалуйста, выберете изображение, содержащее QR код с информацией о чеке + + + Picture was not selected + Изображение не было выбрано + + + Please, select a picture to scan + Пожалуйста, выберете изображение для сканирования + + + + Form + Форма + + + + Optical Character Recognition + Оптическое распознавание символов + + + + Text from E-Mail + Текст из электронного письма + + + + OCRScene + + + Form + Форма + + + + Choose + Выбрать + + + + Path to image: + Путь к изображению: + + + + Store: + Магазин: + + + + Recognized text will be shown below as soon as image will be processed. Please, edit it + Распознанный текст будет показан ниже как только изображение обработается. Пожалуйста, отредактируйте + + + + Back + Назад + + + + Parse + Парсить + + + + An error has occured. Check was matched incorrectly. Vector sizes are different. Please, contact the developer. + Произошла ошибка. Чек был прочитан неверно. Размеры векторов различаются. Пожалуйста, сообщите об этом разработчику. + + + Error in parsing Ошибка в парсинге - - Please, select a picture where QR code that contains info about check is present - Пожалуйста, выберете изображение, содержащее QR код с информацией о чеке + + Please, select a picture to scan + Пожалуйста, выберете изображение для сканирования - - + Picture was not selected Изображение не было выбрано - - Please, select a picture to scan - Пожалуйста, выберете изображение для сканирования + + Path to image: + Путь к изображению: + + + + OFDScene + + + Form + Форма + + + + Total + Итого + + + + Back + Назад + + + + or + или + + + + FD (Fiscal Document) + ФД + + + + Date and time of purchase + Дата и время покупки + + + + Funds income + Приход средств + + + + Funds return + Возврат средств + + + + Funds spend + Расход средств + + + + Spends return + Возврат расхода + + + + Use your phone as a QR code scanner + Использовать телефон как сканнер QR + + + + FN (Fiscal Number) + ФН + + + + FI (Fiscal Identifier) + ФП + + + + Choose image on your PC + Выбрать изображение на компьютере + + + + Operation type + Тип операции + + + + Parse + Парсить + + + + Please, select a picture where QR code that contains info about check is present + Пожалуйста, выберете изображение, содержащее QR код с информацией о чеке + + + + Picture was not selected + Изображение не было выбрано + + + + Selected image: + Выбранное изображение: + + + + Captcha was not solved correctly! + Капча была решена неверно! + + + + Captcha is incorrect + Капча введена неверно + + + + Internal server error. Please, try again later. + Внутренняя ошибка сервера. Пожалуйста, попробуйте снова позже. + + + + Internal server error + Внутренняя ошибка сервера + + + + Check not found. Please, ensure correctness of entered data. + Чек не найден. Пожалуйста, убедитесь в правильности введённых данных. + + + + Check was not found + Чек не найден OutputDialog - + Dialog Диалог - + Path to export: Путь для экспорта: - + Choose Выбрать - + Print header Печатать заголовок - + Goods name Имя товара - + Goods price Цена товара - + Goods quantity Количество товара - + Goods net weight Масса нетто товара - + Goods total Всего за товар - + position позиция - + name алиас - + 1 1 - + Name Имя - + 2 2 - + Price Цена - + 3 3 - + Quantity Количество - + 4 4 - + Net weight Масса нетто - + 5 5 - + Total price Всего - + Print total Печатать Итого @@ -317,20 +528,18 @@ SettingsDialog - You need to restart program to apply language changes - Требуется перезагрузить программу, чтобы применить изменения языка + Требуется перезагрузить программу, чтобы применить изменения языка - Restart required - Требуется перезагрузка + Требуется перезагрузка SolveCaptchaDialog - + Dialog Диалог @@ -348,110 +557,88 @@ settingsdialog - Dialog - Диалог + Диалог - Goods name position - Позиция имени товара + Позиция имени товара - Goods price per unit alias - Алиас цены товара + Алиас цены товара - TextLabel - Язык + Язык - en_US - en_US + en_US - ru_RU - ru_RU + ru_RU - - Choose - Выбрать + Выбрать - Print header - Печатать заголовок + Печатать заголовок - Goods net weight alias - Алиас массы нетто товара + Алиас массы нетто товара - Stores modules url - URL модулей магазина + URL модулей магазина - Goods total alias - Алиас всего за продукт + Алиас всего за продукт - Goods name alias - Алиас имени товара + Алиас имени товара - Goods quantity alias - Алиас количества товара + Алиас количества товара - Stores modules directory - Директория модулей магазина + Директория модулей магазина - OFD modules directory - Директория модулей ОФД + Директория модулей ОФД - Goods price per unit position - Позиция центы товара + Позиция центы товара - Goods net weight position - Позиция массы нетто товара + Позиция массы нетто товара - OFD modules url - URL модулей ОФД + URL модулей ОФД - Goods total position - Позиция всего за товар + Позиция всего за товар - Goods quantity position - Позиция количества товара + Позиция количества товара - Print total - Печатать Итого + Печатать Итого diff --git a/utils/utils.cpp b/utils/utils.cpp index e220616..ec5ab3d 100644 --- a/utils/utils.cpp +++ b/utils/utils.cpp @@ -1,5 +1,6 @@ #include "utils.h" +#include #include #include #include @@ -7,6 +8,37 @@ #include #include #include "../exceptions/ofdrequestexception.h" +#include "settings/settings.h" +#include +#include +#include +#include + +std::string get_local_ip_address() { + struct ifaddrs * ifAddrStruct=NULL; + struct ifaddrs * ifa=NULL; + void * tmpAddrPtr=NULL; + + getifaddrs(&ifAddrStruct); + + for (ifa = ifAddrStruct; ifa != NULL; ifa = ifa->ifa_next) { + if (ifa->ifa_addr == nullptr) continue; + if (ifa->ifa_addr->sa_family==AF_INET) { + tmpAddrPtr=&((struct sockaddr_in *)ifa->ifa_addr)->sin_addr; + char addressBuffer[128]; + inet_ntop(AF_INET, tmpAddrPtr, addressBuffer, INET_ADDRSTRLEN); + + std::string value(addressBuffer); + if (!strncmp(value.c_str(), "192.168", 7)) { + return value; + } + } + } + if (ifAddrStruct!=NULL) + freeifaddrs(ifAddrStruct); + + throw std::runtime_error(QWidget::tr("Could not find any usable local IP address. If you beleive that this is problem with the program, please, contact the developer.").toStdString()); +} std::string to_utf8(std::wstring wide_string) { static std::wstring_convert> utf8_conv; @@ -94,8 +126,9 @@ std::vector find_in_html(std::string& html, std::string regex, std it != end; it++) { std::wstring found_entry = from_utf8(it->str()); + std::wcout << "Found: " << found_entry << std::endl; std::wstring extracted = substring_from_to(found_entry, from_utf8(html_start), from_utf8(html_end)); - + std::wcout << "Extracted: " << extracted << std::endl; parsed.push_back(extracted); } return parsed; @@ -106,11 +139,42 @@ std::vector find_products_in_html(std::string html) { } std::vector find_amounts_in_html(std::string html) { - return find_in_html(html, "\\d+<\\/span>", "", "<\\/span>"); + std::vector founds = find_in_html(html, "
\\d+(\\.|\\,)?\\d{0,3}<\\/span>", "", "<\\/span>"); + for (auto &found : founds) { + std::replace(found.begin(), found.end(), ',', '.'); + } + + return founds; } std::vector find_prices_in_html(std::string html) { - return find_in_html(html, "X <\\/span>\\d+\\.\\d{2}<\\/span>", "X <\\/span>", "<\\/span>"); + std::vector founds = find_in_html(html, "X <\\/span>\\d+(\\.|,)\\d{2}<\\/span>", "X <\\/span>", "<\\/span>"); + for (auto &found : founds) { + std::replace(found.begin(), found.end(), ',', '.'); + } + + return founds; +} + +void dumpVectorsToStderr(std::vector &products, std::vector &amounts, std::vector &prices) { + std::cerr << "Products: "; + for (auto &product : products) { + std::cerr << to_utf8(product) << "|[]|"; + } + std::cerr << std::endl; + + std::cerr << "Amounts: "; + for (auto &amount : amounts) { + std::wcerr << amount << " "; + } + std::cerr << std::endl; + + std::cerr << "Prices: "; + for (auto &price : prices) { + std::wcerr << price << " "; + } + + std::cerr << std::endl; } Check parseOfdRuAnswer(std::string html) { @@ -133,6 +197,10 @@ Check parseOfdRuAnswer(std::string html) { } if ((products.size() + amounts.size() + prices.size())/products.size() != 3) { + dumpVectorsToStderr(products, amounts, prices); + //TOOD: make new setting "app_home" and get all path using it. + std::ofstream error_log(get_path_relative_to_home(".local/share/checks_parser/error_log.txt"), std::ios_base::app); + error_log << trimmed << std::endl; std::cerr << "An error has occured during the parsing of html. Please, contact the developer." << std::endl; std::exit(-1); } diff --git a/utils/utils.h b/utils/utils.h index a235f28..7bb7402 100644 --- a/utils/utils.h +++ b/utils/utils.h @@ -5,6 +5,7 @@ #include #include "../check/check.h" +std::string get_local_ip_address(); std::string to_utf8(std::wstring wide_string); std::wstring from_utf8(std::string string);