#include "assets/image_format.h" #include "assets/image_metadata.h" #include "assets/ppi_header.h" #include "document/document.h" #include "foundation/parse.h" #include "foundation/result.h" #include "paint/stroke.h" #include "ui_core/layout_xml.h" #include #include #include #include #include #include #include #include namespace { struct DocumentArgs { std::uint32_t width = 0; std::uint32_t height = 0; std::uint32_t layers = 1; std::uint32_t frames = 1; std::uint32_t frame_duration_ms = 100; }; struct InspectImageArgs { std::string path; }; struct ParseLayoutArgs { std::string path; }; struct InspectProjectArgs { std::string path; }; struct SimulateStrokeArgs { std::uint32_t x1 = 0; std::uint32_t y1 = 0; std::uint32_t x2 = 0; std::uint32_t y2 = 0; std::uint32_t spacing = 1; }; void print_error(std::string_view command, std::string_view message) { std::cout << "{\"ok\":false,\"command\":\"" << command << "\",\"error\":\"" << message << "\"}\n"; } void print_help() { std::cout << "pano_cli commands:\n" << " create-document --width N --height N [--layers N] [--frames N] [--frame-duration-ms N]\n" << " inspect-image --path FILE\n" << " inspect-project --path FILE\n" << " parse-layout --path FILE\n" << " simulate-stroke --x1 N --y1 N --x2 N --y2 N [--spacing N]\n" << " --help\n"; } pp::foundation::Status parse_document_args(int argc, char** argv, DocumentArgs& args) { for (int i = 2; i < argc; ++i) { const std::string_view key(argv[i]); if (key == "--width" || key == "--height" || key == "--layers" || key == "--frames" || key == "--frame-duration-ms") { if (i + 1 >= argc) { return pp::foundation::Status::invalid_argument("missing value for option"); } const auto value = pp::foundation::parse_u32(argv[++i]); if (!value) { return value.status(); } if (key == "--width") { args.width = value.value(); } else if (key == "--height") { args.height = value.value(); } else if (key == "--layers") { args.layers = value.value(); } else if (key == "--frames") { args.frames = value.value(); } else { args.frame_duration_ms = value.value(); } } else { return pp::foundation::Status::invalid_argument("unknown option"); } } if (args.width == 0 || args.height == 0) { return pp::foundation::Status::invalid_argument("width and height must be greater than zero"); } if (args.layers == 0) { return pp::foundation::Status::invalid_argument("layer count must be greater than zero"); } if (args.frames == 0) { return pp::foundation::Status::invalid_argument("frame count must be greater than zero"); } if (args.frame_duration_ms == 0) { return pp::foundation::Status::invalid_argument("frame duration must be greater than zero"); } return pp::foundation::Status::success(); } int create_document(int argc, char** argv) { DocumentArgs args; const auto status = parse_document_args(argc, argv, args); if (!status.ok()) { print_error("create-document", status.message); return 2; } const auto document_result = pp::document::CanvasDocument::create( pp::document::DocumentConfig { .width = args.width, .height = args.height, .layer_count = args.layers, }); if (!document_result) { print_error("create-document", document_result.status().message); return 2; } auto document = document_result.value(); const auto duration_status = document.set_frame_duration(0, args.frame_duration_ms); if (!duration_status.ok()) { print_error("create-document", duration_status.message); return 2; } for (std::uint32_t i = 1; i < args.frames; ++i) { const auto added_frame = document.add_frame(args.frame_duration_ms); if (!added_frame) { print_error("create-document", added_frame.status().message); return 2; } } std::cout << "{\"ok\":true,\"command\":\"create-document\",\"document\":{" << "\"width\":" << document.width() << ",\"height\":" << document.height() << ",\"layers\":" << document.layers().size() << ",\"activeLayer\":" << document.active_layer_index() << ",\"frames\":" << document.frames().size() << ",\"activeFrame\":" << document.active_frame_index() << ",\"animationDurationMs\":" << document.animation_duration_ms() << "}}\n"; return 0; } pp::foundation::Status parse_inspect_image_args(int argc, char** argv, InspectImageArgs& args) { for (int i = 2; i < argc; ++i) { const std::string_view key(argv[i]); if (key == "--path") { if (i + 1 >= argc) { return pp::foundation::Status::invalid_argument("missing value for option"); } args.path = argv[++i]; } else { return pp::foundation::Status::invalid_argument("unknown option"); } } if (args.path.empty()) { return pp::foundation::Status::invalid_argument("path must not be empty"); } return pp::foundation::Status::success(); } int inspect_image(int argc, char** argv) { InspectImageArgs args; const auto status = parse_inspect_image_args(argc, argv, args); if (!status.ok()) { print_error("inspect-image", status.message); return 2; } std::ifstream stream(args.path, std::ios::binary); if (!stream) { print_error("inspect-image", "image file could not be opened"); return 2; } const std::vector chars { std::istreambuf_iterator(stream), std::istreambuf_iterator() }; const auto* data = reinterpret_cast(chars.data()); const auto format = pp::assets::detect_image_format(std::span(data, chars.size())); if (!format) { print_error("inspect-image", format.status().message); return 2; } pp::foundation::Result metadata = pp::foundation::Result::failure( pp::foundation::Status::invalid_argument("image metadata is unavailable")); if (format.value() == pp::assets::ImageFormat::png) { metadata = pp::assets::parse_png_metadata(std::span(data, chars.size())); if (!metadata) { print_error("inspect-image", metadata.status().message); return 2; } } std::cout << "{\"ok\":true,\"command\":\"inspect-image\",\"format\":\"" << pp::assets::image_format_name(format.value()) << "\",\"bytes\":" << chars.size(); if (metadata) { std::cout << ",\"metadata\":{\"width\":" << metadata.value().width << ",\"height\":" << metadata.value().height << ",\"bitDepth\":" << static_cast(metadata.value().bit_depth) << ",\"components\":" << static_cast(metadata.value().components) << ",\"colorType\":\"" << pp::assets::image_color_type_name(metadata.value().color_type) << "\"}"; } else { std::cout << ",\"metadata\":null"; } std::cout << "}\n"; return 0; } pp::foundation::Status parse_inspect_project_args(int argc, char** argv, InspectProjectArgs& args) { for (int i = 2; i < argc; ++i) { const std::string_view key(argv[i]); if (key == "--path") { if (i + 1 >= argc) { return pp::foundation::Status::invalid_argument("missing value for option"); } args.path = argv[++i]; } else { return pp::foundation::Status::invalid_argument("unknown option"); } } if (args.path.empty()) { return pp::foundation::Status::invalid_argument("path must not be empty"); } return pp::foundation::Status::success(); } int inspect_project(int argc, char** argv) { InspectProjectArgs args; const auto status = parse_inspect_project_args(argc, argv, args); if (!status.ok()) { print_error("inspect-project", status.message); return 2; } std::ifstream stream(args.path, std::ios::binary); if (!stream) { print_error("inspect-project", "project file could not be opened"); return 2; } const std::vector chars { std::istreambuf_iterator(stream), std::istreambuf_iterator() }; const auto* data = reinterpret_cast(chars.data()); const auto header = pp::assets::parse_ppi_header(std::span(data, chars.size())); if (!header) { print_error("inspect-project", header.status().message); return 2; } std::cout << "{\"ok\":true,\"command\":\"inspect-project\"" << ",\"documentVersion\":\"" << header.value().document_version.major << "." << header.value().document_version.minor << "\"" << ",\"softwareVersion\":\"" << header.value().software_version.major << "." << header.value().software_version.minor << "." << header.value().software_version.fix << "." << header.value().software_version.build << "\"" << ",\"thumbnail\":{\"width\":" << header.value().thumbnail.width << ",\"height\":" << header.value().thumbnail.height << ",\"components\":" << header.value().thumbnail.components << "}}\n"; return 0; } pp::foundation::Status parse_simulate_stroke_args(int argc, char** argv, SimulateStrokeArgs& args) { for (int i = 2; i < argc; ++i) { const std::string_view key(argv[i]); if (key == "--x1" || key == "--y1" || key == "--x2" || key == "--y2" || key == "--spacing") { if (i + 1 >= argc) { return pp::foundation::Status::invalid_argument("missing value for option"); } const auto value = pp::foundation::parse_u32(argv[++i]); if (!value) { return value.status(); } if (key == "--x1") { args.x1 = value.value(); } else if (key == "--y1") { args.y1 = value.value(); } else if (key == "--x2") { args.x2 = value.value(); } else if (key == "--y2") { args.y2 = value.value(); } else { args.spacing = value.value(); } } else { return pp::foundation::Status::invalid_argument("unknown option"); } } if (args.spacing == 0) { return pp::foundation::Status::invalid_argument("stroke spacing must be greater than zero"); } return pp::foundation::Status::success(); } int simulate_stroke(int argc, char** argv) { SimulateStrokeArgs args; const auto status = parse_simulate_stroke_args(argc, argv, args); if (!status.ok()) { print_error("simulate-stroke", status.message); return 2; } const pp::paint::StrokePoint points[] { pp::paint::StrokePoint { .x = static_cast(args.x1), .y = static_cast(args.y1), .pressure = 1.0F, }, pp::paint::StrokePoint { .x = static_cast(args.x2), .y = static_cast(args.y2), .pressure = 1.0F, }, }; const auto samples = pp::paint::sample_stroke( points, pp::paint::StrokeSamplingConfig { .spacing = static_cast(args.spacing), }); if (!samples) { print_error("simulate-stroke", samples.status().message); return 2; } const auto& first = samples.value().front(); const auto& last = samples.value().back(); std::cout << "{\"ok\":true,\"command\":\"simulate-stroke\"" << ",\"samples\":" << samples.value().size() << ",\"first\":{\"x\":" << first.x << ",\"y\":" << first.y << "}" << ",\"last\":{\"x\":" << last.x << ",\"y\":" << last.y << ",\"distance\":" << last.distance << "}}\n"; return 0; } pp::foundation::Status parse_layout_args(int argc, char** argv, ParseLayoutArgs& args) { for (int i = 2; i < argc; ++i) { const std::string_view key(argv[i]); if (key == "--path") { if (i + 1 >= argc) { return pp::foundation::Status::invalid_argument("missing value for option"); } args.path = argv[++i]; } else { return pp::foundation::Status::invalid_argument("unknown option"); } } if (args.path.empty()) { return pp::foundation::Status::invalid_argument("path must not be empty"); } return pp::foundation::Status::success(); } int parse_layout(int argc, char** argv) { ParseLayoutArgs args; const auto status = parse_layout_args(argc, argv, args); if (!status.ok()) { print_error("parse-layout", status.message); return 2; } std::ifstream stream(args.path, std::ios::binary); if (!stream) { print_error("parse-layout", "layout file could not be opened"); return 2; } const std::string xml { std::istreambuf_iterator(stream), std::istreambuf_iterator() }; const auto summary = pp::ui::parse_layout_xml(xml); if (!summary) { print_error("parse-layout", summary.status().message); return 2; } std::cout << "{\"ok\":true,\"command\":\"parse-layout\"" << ",\"nodes\":" << summary.value().node_count << ",\"lengthAttributes\":" << summary.value().length_attribute_count << "}\n"; return 0; } } int main(int argc, char** argv) { if (argc < 2) { print_help(); return 1; } const std::string_view command(argv[1]); if (command == "--help" || command == "-h") { print_help(); return 0; } if (command == "create-document") { return create_document(argc, argv); } if (command == "inspect-image") { return inspect_image(argc, argv); } if (command == "inspect-project") { return inspect_project(argc, argv); } if (command == "simulate-stroke") { return simulate_stroke(argc, argv); } if (command == "parse-layout") { return parse_layout(argc, argv); } print_error(command, "unknown command"); return 2; }