Compare commits
2 Commits
69810e1c5c
...
f8a008a105
| Author | SHA1 | Date | |
|---|---|---|---|
|
f8a008a105
|
|||
|
81b0f25f32
|
@@ -21,7 +21,7 @@
|
||||
#include <string>
|
||||
#include <vector>
|
||||
|
||||
#include "interface/IPrinterDriver.hpp"
|
||||
#include "printers/interface/IPrinterDriver.hpp"
|
||||
#include "libusbwrap/LibUsbTypes.hpp"
|
||||
|
||||
namespace ptprnt {
|
||||
|
||||
@@ -25,16 +25,21 @@
|
||||
|
||||
#include "PrinterDriverFactory.hpp"
|
||||
#include "cli/CliParser.hpp"
|
||||
#include "cli/interface/ICliParser.hpp"
|
||||
#include "constants.hpp"
|
||||
#include "core/PrinterService.hpp"
|
||||
#include "core/interface/IPrinterService.hpp"
|
||||
#include "graphics/LabelBuilder.hpp"
|
||||
|
||||
namespace ptprnt {
|
||||
|
||||
PtouchPrint::PtouchPrint(const char* versionString)
|
||||
: mVersionString(versionString),
|
||||
mCliParser(std::make_unique<cli::CliParser>(ptprnt::APP_DESC, versionString)),
|
||||
mPrinterService(std::make_unique<core::PrinterService>()) {}
|
||||
: PtouchPrint(versionString, std::make_unique<cli::CliParser>(ptprnt::APP_DESC, versionString),
|
||||
std::make_unique<core::PrinterService>()) {}
|
||||
|
||||
PtouchPrint::PtouchPrint(const char* versionString, std::unique_ptr<cli::ICliParser> cliParser,
|
||||
std::unique_ptr<core::IPrinterService> printerService)
|
||||
: mVersionString(versionString), mCliParser(std::move(cliParser)), mPrinterService(std::move(printerService)) {}
|
||||
|
||||
PtouchPrint::~PtouchPrint() = default;
|
||||
|
||||
@@ -97,7 +102,7 @@ void PtouchPrint::setupLogger() {
|
||||
|
||||
bool PtouchPrint::handleListDrivers() {
|
||||
auto driverFactory = std::make_unique<PrinterDriverFactory>();
|
||||
auto drivers = driverFactory->listAllDrivers();
|
||||
auto drivers = driverFactory->listAllDrivers();
|
||||
|
||||
fmt::print("Available printer drivers:\n");
|
||||
for (const auto& driver : drivers) {
|
||||
@@ -133,7 +138,7 @@ bool PtouchPrint::handlePrinting() {
|
||||
for (const auto& [cmdType, value] : options.commands) {
|
||||
switch (cmdType) {
|
||||
case cli::CommandType::Text:
|
||||
labelBuilder.addText(value + "\n");
|
||||
labelBuilder.addText(value);
|
||||
break;
|
||||
case cli::CommandType::Font:
|
||||
spdlog::debug("Setting font to {}", value);
|
||||
|
||||
@@ -23,11 +23,11 @@
|
||||
#include <string>
|
||||
|
||||
namespace ptprnt::cli {
|
||||
class CliParser;
|
||||
class ICliParser;
|
||||
}
|
||||
|
||||
namespace ptprnt::core {
|
||||
class PrinterService;
|
||||
class IPrinterService;
|
||||
}
|
||||
|
||||
namespace ptprnt {
|
||||
@@ -37,14 +37,27 @@ namespace ptprnt {
|
||||
*
|
||||
* Acts as a thin glue layer coordinating CLI parsing and core printer functionality.
|
||||
* Separates CLI frontend concerns from the core library.
|
||||
*
|
||||
* Uses interfaces (ICliParser, IPrinterService) to enable dependency injection
|
||||
* and facilitate unit testing with mocks.
|
||||
*/
|
||||
class PtouchPrint {
|
||||
public:
|
||||
/**
|
||||
* @brief Construct the application
|
||||
* @brief Construct the application with default implementations
|
||||
* @param versionString Version string to display
|
||||
*/
|
||||
PtouchPrint(const char* versionString);
|
||||
|
||||
/**
|
||||
* @brief Construct with custom implementations (for testing)
|
||||
* @param versionString Version string to display
|
||||
* @param cliParser Custom CLI parser implementation
|
||||
* @param printerService Custom printer service implementation
|
||||
*/
|
||||
PtouchPrint(const char* versionString, std::unique_ptr<cli::ICliParser> cliParser,
|
||||
std::unique_ptr<core::IPrinterService> printerService);
|
||||
|
||||
~PtouchPrint(); // Must be defined in .cpp where complete types are visible
|
||||
|
||||
// This is basically a singleton application class, no need to copy or move
|
||||
@@ -73,8 +86,8 @@ class PtouchPrint {
|
||||
bool handlePrinting();
|
||||
|
||||
std::string mVersionString;
|
||||
std::unique_ptr<cli::CliParser> mCliParser;
|
||||
std::unique_ptr<core::PrinterService> mPrinterService;
|
||||
std::unique_ptr<cli::ICliParser> mCliParser;
|
||||
std::unique_ptr<core::IPrinterService> mPrinterService;
|
||||
};
|
||||
|
||||
} // namespace ptprnt
|
||||
|
||||
@@ -23,8 +23,8 @@
|
||||
|
||||
namespace ptprnt::cli {
|
||||
|
||||
CliParser::CliParser(const std::string& appDescription, std::string versionString)
|
||||
: mApp(appDescription), mVersionString(std::move(versionString)) {
|
||||
CliParser::CliParser(std::string appDescription, std::string versionString)
|
||||
: mApp(std::move(appDescription)), mVersionString(std::move(versionString)) {
|
||||
setupParser();
|
||||
}
|
||||
|
||||
@@ -67,8 +67,9 @@ void CliParser::setupParser() {
|
||||
|
||||
// Text printing options
|
||||
// Note: CLI11 options are processed in order when using ->each() with callbacks
|
||||
mApp.add_option("-t,--text",
|
||||
"Text to print (can be used multiple times, use formatting options before to influence text layout)")
|
||||
mApp.add_option(
|
||||
"-t,--text",
|
||||
"Text to print (can be used multiple times, use formatting options before to influence text layout)")
|
||||
->multi_option_policy(CLI::MultiOptionPolicy::TakeAll)
|
||||
->each([this](const std::string& text) { mOptions.commands.emplace_back(CommandType::Text, text); });
|
||||
|
||||
|
||||
@@ -22,47 +22,28 @@
|
||||
#include <CLI/CLI.hpp>
|
||||
|
||||
#include <string>
|
||||
#include <vector>
|
||||
|
||||
#include "interface/ICliParser.hpp"
|
||||
|
||||
namespace ptprnt::cli {
|
||||
|
||||
/**
|
||||
* @brief Types of CLI commands that can be issued
|
||||
*/
|
||||
enum class CommandType { None = 0, Text = 1, FontSize = 2, Font = 3, VAlign = 4, HAlign = 5 };
|
||||
|
||||
/**
|
||||
* @brief A command with its type and value
|
||||
*/
|
||||
using Command = std::pair<CommandType, std::string>;
|
||||
|
||||
/**
|
||||
* @brief Parsed CLI options and commands
|
||||
*/
|
||||
struct CliOptions {
|
||||
bool verbose{false};
|
||||
bool trace{false};
|
||||
bool listDrivers{false};
|
||||
std::string printerSelection{"auto"};
|
||||
std::vector<Command> commands{};
|
||||
};
|
||||
|
||||
/**
|
||||
* @brief CLI argument parser for ptprnt
|
||||
*
|
||||
* Handles all command-line argument parsing using CLI11.
|
||||
* Concrete implementation of ICliParser using CLI11.
|
||||
* Handles all command-line argument parsing.
|
||||
* Separates CLI concerns from core library functionality.
|
||||
*/
|
||||
class CliParser {
|
||||
class CliParser : public ICliParser {
|
||||
public:
|
||||
/**
|
||||
* @brief Construct a CLI parser
|
||||
* @param appDescription Application description for help text
|
||||
* @param versionString Version string to display
|
||||
*/
|
||||
CliParser(const std::string& appDescription, std::string versionString);
|
||||
CliParser(std::string appDescription, std::string versionString);
|
||||
|
||||
~CliParser() = default;
|
||||
~CliParser() override = default;
|
||||
|
||||
CliParser(const CliParser&) = delete;
|
||||
CliParser& operator=(const CliParser&) = delete;
|
||||
@@ -75,13 +56,13 @@ class CliParser {
|
||||
* @param argv Argument values
|
||||
* @return 0 on success, positive value if should exit immediately (help/version), negative on error
|
||||
*/
|
||||
int parse(int argc, char** argv);
|
||||
int parse(int argc, char** argv) override;
|
||||
|
||||
/**
|
||||
* @brief Get the parsed options
|
||||
* @return Reference to parsed options
|
||||
*/
|
||||
[[nodiscard]] const CliOptions& getOptions() const { return mOptions; }
|
||||
[[nodiscard]] const CliOptions& getOptions() const override { return mOptions; }
|
||||
|
||||
private:
|
||||
void setupParser();
|
||||
|
||||
73
src/cli/interface/ICliParser.hpp
Normal file
73
src/cli/interface/ICliParser.hpp
Normal file
@@ -0,0 +1,73 @@
|
||||
/*
|
||||
ptrnt - print labels on linux
|
||||
Copyright (C) 2025 Moritz Martinius
|
||||
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
#include <string>
|
||||
#include <vector>
|
||||
|
||||
namespace ptprnt::cli {
|
||||
|
||||
/**
|
||||
* @brief Types of CLI commands that can be issued
|
||||
*/
|
||||
enum class CommandType { None = 0, Text = 1, FontSize = 2, Font = 3, VAlign = 4, HAlign = 5 };
|
||||
|
||||
/**
|
||||
* @brief A command with its type and value
|
||||
*/
|
||||
using Command = std::pair<CommandType, std::string>;
|
||||
|
||||
/**
|
||||
* @brief Parsed CLI options and commands
|
||||
*/
|
||||
struct CliOptions {
|
||||
bool verbose{false};
|
||||
bool trace{false};
|
||||
bool listDrivers{false};
|
||||
std::string printerSelection{"auto"};
|
||||
std::vector<Command> commands{};
|
||||
};
|
||||
|
||||
/**
|
||||
* @brief Interface for CLI argument parsing
|
||||
*
|
||||
* This interface allows for mocking CLI parsing in unit tests
|
||||
* and provides a clear contract for CLI parser implementations.
|
||||
*/
|
||||
class ICliParser {
|
||||
public:
|
||||
virtual ~ICliParser() = default;
|
||||
|
||||
/**
|
||||
* @brief Parse command line arguments
|
||||
* @param argc Argument count
|
||||
* @param argv Argument values
|
||||
* @return 0 on success, positive value if should exit immediately (help/version), negative on error
|
||||
*/
|
||||
virtual int parse(int argc, char** argv) = 0;
|
||||
|
||||
/**
|
||||
* @brief Get the parsed options
|
||||
* @return Reference to parsed options
|
||||
*/
|
||||
[[nodiscard]] virtual const CliOptions& getOptions() const = 0;
|
||||
};
|
||||
|
||||
} // namespace ptprnt::cli
|
||||
@@ -54,6 +54,20 @@ std::vector<std::shared_ptr<IPrinterDriver>> PrinterService::detectPrinters() {
|
||||
}
|
||||
|
||||
std::shared_ptr<IPrinterDriver> PrinterService::selectPrinter(const std::string& printerName) {
|
||||
// If a specific printer is requested by name (not "auto"), try to create it directly
|
||||
if (printerName != "auto") {
|
||||
auto driverFactory = std::make_unique<PrinterDriverFactory>();
|
||||
auto printer = driverFactory->createByName(printerName);
|
||||
if (printer) {
|
||||
mCurrentPrinter = printer;
|
||||
spdlog::info("Using explicitly selected printer: {}", printerName);
|
||||
return mCurrentPrinter;
|
||||
}
|
||||
spdlog::error("Printer driver '{}' not found", printerName);
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
// Auto mode: detect USB printers
|
||||
if (mDetectedPrinters.empty()) {
|
||||
detectPrinters();
|
||||
}
|
||||
@@ -63,24 +77,10 @@ std::shared_ptr<IPrinterDriver> PrinterService::selectPrinter(const std::string&
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
// Auto-select first printer
|
||||
if (printerName == "auto") {
|
||||
mCurrentPrinter = mDetectedPrinters.front();
|
||||
spdlog::info("Auto-selected printer: {}", mCurrentPrinter->getName());
|
||||
return mCurrentPrinter;
|
||||
}
|
||||
|
||||
// Select printer by name
|
||||
for (auto& printer : mDetectedPrinters) {
|
||||
if (printer->getDriverName() == printerName) {
|
||||
mCurrentPrinter = printer;
|
||||
spdlog::info("Using explicitly selected printer: {}", printerName);
|
||||
return mCurrentPrinter;
|
||||
}
|
||||
}
|
||||
|
||||
spdlog::error("Printer '{}' not found", printerName);
|
||||
return nullptr;
|
||||
// Auto-select first detected printer
|
||||
mCurrentPrinter = mDetectedPrinters.front();
|
||||
spdlog::info("Auto-selected printer: {}", mCurrentPrinter->getName());
|
||||
return mCurrentPrinter;
|
||||
}
|
||||
|
||||
bool PrinterService::printLabel(std::unique_ptr<graphics::ILabel> label) {
|
||||
|
||||
@@ -23,23 +23,25 @@
|
||||
#include <string>
|
||||
#include <vector>
|
||||
|
||||
#include "interface/IPrinterDriver.hpp"
|
||||
#include "libusbwrap/UsbDeviceFactory.hpp"
|
||||
#include "../printers/interface/IPrinterDriver.hpp"
|
||||
#include "interface/IPrinterService.hpp"
|
||||
#include "../libusbwrap/UsbDeviceFactory.hpp"
|
||||
|
||||
namespace ptprnt::core {
|
||||
|
||||
/**
|
||||
* @brief Core service for printer operations
|
||||
*
|
||||
* Concrete implementation of IPrinterService.
|
||||
* Provides the core library functionality for:
|
||||
* - Detecting printers
|
||||
* - Selecting printers
|
||||
* - Building and printing labels
|
||||
*/
|
||||
class PrinterService {
|
||||
class PrinterService : public IPrinterService {
|
||||
public:
|
||||
PrinterService();
|
||||
~PrinterService() = default;
|
||||
~PrinterService() override = default;
|
||||
|
||||
PrinterService(const PrinterService&) = delete;
|
||||
PrinterService& operator=(const PrinterService&) = delete;
|
||||
@@ -50,33 +52,33 @@ class PrinterService {
|
||||
* @brief Initialize USB device factory
|
||||
* @return true on success, false on failure
|
||||
*/
|
||||
bool initialize();
|
||||
bool initialize() override;
|
||||
|
||||
/**
|
||||
* @brief Detect all compatible printers
|
||||
* @return Vector of detected printers
|
||||
*/
|
||||
std::vector<std::shared_ptr<IPrinterDriver>> detectPrinters();
|
||||
std::vector<std::shared_ptr<IPrinterDriver>> detectPrinters() override;
|
||||
|
||||
/**
|
||||
* @brief Select a printer by name or auto-detect
|
||||
* @param printerName Printer driver name, or "auto" for first detected
|
||||
* @return Printer driver, or nullptr if not found
|
||||
*/
|
||||
std::shared_ptr<IPrinterDriver> selectPrinter(const std::string& printerName);
|
||||
std::shared_ptr<IPrinterDriver> selectPrinter(const std::string& printerName) override;
|
||||
|
||||
/**
|
||||
* @brief Get the currently selected printer
|
||||
* @return Current printer, or nullptr if none selected
|
||||
*/
|
||||
[[nodiscard]] std::shared_ptr<IPrinterDriver> getCurrentPrinter() const { return mCurrentPrinter; }
|
||||
[[nodiscard]] std::shared_ptr<IPrinterDriver> getCurrentPrinter() const override { return mCurrentPrinter; }
|
||||
|
||||
/**
|
||||
* @brief Print a label
|
||||
* @param label The label to print
|
||||
* @return true on success, false on failure
|
||||
*/
|
||||
bool printLabel(std::unique_ptr<graphics::ILabel> label);
|
||||
bool printLabel(std::unique_ptr<graphics::ILabel> label) override;
|
||||
|
||||
private:
|
||||
libusbwrap::UsbDeviceFactory mUsbDeviceFactory;
|
||||
|
||||
74
src/core/interface/IPrinterService.hpp
Normal file
74
src/core/interface/IPrinterService.hpp
Normal file
@@ -0,0 +1,74 @@
|
||||
/*
|
||||
ptrnt - print labels on linux
|
||||
Copyright (C) 2025 Moritz Martinius
|
||||
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
#include <memory>
|
||||
#include <string>
|
||||
#include <vector>
|
||||
|
||||
#include "../../printers/interface/IPrinterDriver.hpp"
|
||||
#include "../../graphics/interface/ILabel.hpp"
|
||||
|
||||
namespace ptprnt::core {
|
||||
|
||||
/**
|
||||
* @brief Interface for core printer service operations
|
||||
*
|
||||
* This interface allows for mocking printer operations in unit tests
|
||||
* and provides a clear contract for printer service implementations.
|
||||
*/
|
||||
class IPrinterService {
|
||||
public:
|
||||
virtual ~IPrinterService() = default;
|
||||
|
||||
/**
|
||||
* @brief Initialize the printer service
|
||||
* @return true on success, false on failure
|
||||
*/
|
||||
virtual bool initialize() = 0;
|
||||
|
||||
/**
|
||||
* @brief Detect all compatible printers
|
||||
* @return Vector of detected printers
|
||||
*/
|
||||
virtual std::vector<std::shared_ptr<IPrinterDriver>> detectPrinters() = 0;
|
||||
|
||||
/**
|
||||
* @brief Select a printer by name or auto-detect
|
||||
* @param printerName Printer driver name, or "auto" for first detected
|
||||
* @return Printer driver, or nullptr if not found
|
||||
*/
|
||||
virtual std::shared_ptr<IPrinterDriver> selectPrinter(const std::string& printerName) = 0;
|
||||
|
||||
/**
|
||||
* @brief Get the currently selected printer
|
||||
* @return Current printer, or nullptr if none selected
|
||||
*/
|
||||
[[nodiscard]] virtual std::shared_ptr<IPrinterDriver> getCurrentPrinter() const = 0;
|
||||
|
||||
/**
|
||||
* @brief Print a label
|
||||
* @param label The label to print
|
||||
* @return true on success, false on failure
|
||||
*/
|
||||
virtual bool printLabel(std::unique_ptr<graphics::ILabel> label) = 0;
|
||||
};
|
||||
|
||||
} // namespace ptprnt::core
|
||||
@@ -22,7 +22,6 @@
|
||||
#include <sys/types.h>
|
||||
|
||||
#include <cstdint>
|
||||
#include <optional>
|
||||
#include <string>
|
||||
#include <vector>
|
||||
|
||||
@@ -44,8 +43,9 @@ enum class Speed {
|
||||
class IUsbDevice {
|
||||
public:
|
||||
virtual ~IUsbDevice() = default;
|
||||
virtual bool open() = 0;
|
||||
virtual void close() = 0;
|
||||
|
||||
virtual bool open() = 0;
|
||||
virtual void close() = 0;
|
||||
|
||||
// libusb wrappers
|
||||
virtual bool detachKernelDriver(int interfaceNo) = 0;
|
||||
|
||||
@@ -23,8 +23,8 @@
|
||||
#include <vector>
|
||||
#include <cstdint>
|
||||
|
||||
#include "../interface/IPrinterDriver.hpp"
|
||||
#include "../interface/IPrinterTypes.hpp"
|
||||
#include "interface/IPrinterDriver.hpp"
|
||||
#include "interface/IPrinterTypes.hpp"
|
||||
#include "../libusbwrap/LibUsbTypes.hpp"
|
||||
#include "../libusbwrap/interface/IUsbDevice.hpp"
|
||||
#include "../graphics/Bitmap.hpp"
|
||||
|
||||
@@ -17,6 +17,8 @@
|
||||
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
#include <spdlog/spdlog.h>
|
||||
#include <sys/types.h>
|
||||
|
||||
@@ -25,10 +27,8 @@
|
||||
|
||||
#include "interface/IPrinterDriver.hpp"
|
||||
#include "interface/IPrinterTypes.hpp"
|
||||
#include "libusbwrap/LibUsbTypes.hpp"
|
||||
#include "libusbwrap/interface/IUsbDevice.hpp"
|
||||
|
||||
#pragma once
|
||||
#include "../libusbwrap/LibUsbTypes.hpp"
|
||||
#include "../libusbwrap/interface/IUsbDevice.hpp"
|
||||
|
||||
namespace ptprnt::printer {
|
||||
namespace p700::commands {
|
||||
|
||||
@@ -22,11 +22,11 @@
|
||||
#include <memory>
|
||||
#include <string_view>
|
||||
|
||||
#include "graphics/Bitmap.hpp"
|
||||
#include "graphics/Monochrome.hpp"
|
||||
#include "graphics/interface/ILabel.hpp"
|
||||
#include "interface/IPrinterTypes.hpp"
|
||||
#include "libusbwrap/interface/IUsbDevice.hpp"
|
||||
#include "../../graphics/Bitmap.hpp"
|
||||
#include "../../graphics/Monochrome.hpp"
|
||||
#include "../../graphics/interface/ILabel.hpp"
|
||||
#include "../../libusbwrap/interface/IUsbDevice.hpp"
|
||||
#include "IPrinterTypes.hpp"
|
||||
|
||||
namespace ptprnt {
|
||||
class IPrinterDriver {
|
||||
Reference in New Issue
Block a user